/* signalproxy.cc
*
* Copyright (C) 2002 The gtkmm Development Team
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library. If not, see .
*/
#include
#include
#include
#include
namespace
{
extern "C"
{
// From functions with C linkage to public static member functions with C++ linkage
static void SignalProxyNormal_slot0_void_callback(GObject* self, void* data)
{
Glib::SignalProxyNormal::slot0_void_callback(self, data);
}
static void SignalProxyConnectionNode_destroy_notify_handler(gpointer data, GClosure* closure)
{
Glib::SignalProxyConnectionNode::destroy_notify_handler(data, closure);
}
} // extern "C"
} // anonymous namespace
namespace Glib
{
// SignalProxyBase implementation:
SignalProxyBase::SignalProxyBase(Glib::ObjectBase* obj) : obj_(obj)
{
}
// SignalProxyNormal implementation:
SignalProxyNormal::SignalProxyNormal(Glib::ObjectBase* obj, const SignalProxyInfo* info)
: SignalProxyBase(obj), info_(info)
{
}
SignalProxyNormal::~SignalProxyNormal() noexcept
{
}
sigc::slot_base&
SignalProxyNormal::connect_impl_(bool notify, const sigc::slot_base& slot, bool after)
{
GCallback c_handler = notify ? info_->notify_callback : info_->callback;
if (c_handler == (GCallback)&slot0_void_callback)
// Callback via a function with C linkage.
c_handler = (GCallback)&SignalProxyNormal_slot0_void_callback;
// create a proxy to hold our connection info
auto pConnectionNode = new SignalProxyConnectionNode(slot, obj_->gobj());
// connect it to glib
// pConnectionNode will be passed in the data argument to the callback.
pConnectionNode->connection_id_ = g_signal_connect_data(obj_->gobj(), info_->signal_name,
c_handler, pConnectionNode, &SignalProxyConnectionNode_destroy_notify_handler,
static_cast(after ? G_CONNECT_AFTER : 0));
return pConnectionNode->slot_;
}
sigc::slot_base&
SignalProxyNormal::connect_impl_(bool notify, sigc::slot_base&& slot, bool after)
{
GCallback c_handler = notify ? info_->notify_callback : info_->callback;
if (c_handler == (GCallback)&slot0_void_callback)
// Callback via a function with C linkage.
c_handler = (GCallback)&SignalProxyNormal_slot0_void_callback;
// create a proxy to hold our connection info
auto pConnectionNode = new SignalProxyConnectionNode(std::move(slot), obj_->gobj());
// connect it to glib
// pConnectionNode will be passed in the data argument to the callback.
pConnectionNode->connection_id_ = g_signal_connect_data(obj_->gobj(), info_->signal_name,
c_handler, pConnectionNode, &SignalProxyConnectionNode_destroy_notify_handler,
static_cast(after ? G_CONNECT_AFTER : 0));
return pConnectionNode->slot_;
}
void
SignalProxyNormal::emission_stop()
{
g_signal_stop_emission_by_name(obj_->gobj(), info_->signal_name);
}
// static
void
SignalProxyNormal::slot0_void_callback(GObject* self, void* data)
{
// Do not try to call a signal on a disassociated wrapper.
if (Glib::ObjectBase::_get_current_wrapper(self))
{
try
{
if (sigc::slot_base* const slot = data_to_slot(data))
(*static_cast*>(slot))();
}
catch (...)
{
Glib::exception_handlers_invoke();
}
}
}
// SignalProxyDetailedBase implementation:
SignalProxyDetailedBase::SignalProxyDetailedBase(
Glib::ObjectBase* obj, const SignalProxyInfo* info, const Glib::ustring& detail_name)
: SignalProxyBase(obj),
info_(info),
detailed_name_(Glib::ustring(info->signal_name) +
(detail_name.empty() ? Glib::ustring() : ("::" + detail_name)))
{
}
SignalProxyDetailedBase::~SignalProxyDetailedBase() noexcept
{
}
sigc::slot_base&
SignalProxyDetailedBase::connect_impl_(bool notify, const sigc::slot_base& slot, bool after)
{
GCallback c_handler = notify ? info_->notify_callback : info_->callback;
if (c_handler == (GCallback)&SignalProxyNormal::slot0_void_callback)
// Callback via a function with C linkage.
c_handler = (GCallback)&SignalProxyNormal_slot0_void_callback;
// create a proxy to hold our connection info
auto pConnectionNode = new SignalProxyConnectionNode(slot, obj_->gobj());
// connect it to glib
// pConnectionNode will be passed in the data argument to the callback.
pConnectionNode->connection_id_ = g_signal_connect_data(obj_->gobj(), detailed_name_.c_str(),
c_handler, pConnectionNode, &SignalProxyConnectionNode_destroy_notify_handler,
static_cast(after ? G_CONNECT_AFTER : 0));
return pConnectionNode->slot_;
}
sigc::slot_base&
SignalProxyDetailedBase::connect_impl_(bool notify, sigc::slot_base&& slot, bool after)
{
GCallback c_handler = notify ? info_->notify_callback : info_->callback;
if (c_handler == (GCallback)&SignalProxyNormal::slot0_void_callback)
// Callback via a function with C linkage.
c_handler = (GCallback)&SignalProxyNormal_slot0_void_callback;
// create a proxy to hold our connection info
auto pConnectionNode = new SignalProxyConnectionNode(std::move(slot), obj_->gobj());
// connect it to glib
// pConnectionNode will be passed in the data argument to the callback.
pConnectionNode->connection_id_ = g_signal_connect_data(obj_->gobj(), detailed_name_.c_str(),
c_handler, pConnectionNode, &SignalProxyConnectionNode_destroy_notify_handler,
static_cast(after ? G_CONNECT_AFTER : 0));
return pConnectionNode->slot_;
}
void
SignalProxyDetailedBase::emission_stop()
{
g_signal_stop_emission_by_name(obj_->gobj(), detailed_name_.c_str());
}
} // namespace Glib