blob: db7b7cb2432d1a5086582c02b092ba3c0c3fe5f0 [file] [log] [blame]
/*
*
* D-Bus++ - C++ bindings for D-Bus
*
* Copyright (C) 2005-2007 Paolo Durante <shackan@gmail.com>
*
*
* 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, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
*/
#ifndef __DBUSXX_INTERFACE_H
#define __DBUSXX_INTERFACE_H
#include <string>
#include <map>
#include "api.h"
#include "util.h"
#include "types.h"
#include "message.h"
namespace DBus {
//todo: this should belong to to properties.h
struct DXXAPI PropertyData
{
bool read;
bool write;
std::string sig;
Variant value;
};
typedef std::map<std::string, PropertyData> PropertyTable;
struct IntrospectedInterface;
class ObjectAdaptor;
class InterfaceAdaptor;
class SignalMessage;
class PendingCall;
typedef std::map<std::string, InterfaceAdaptor *> InterfaceAdaptorTable;
class DXXAPI AdaptorBase
{
public:
virtual const ObjectAdaptor *object() const = 0 ;
protected:
InterfaceAdaptor *find_interface(const std::string &name);
virtual ~AdaptorBase()
{}
virtual void _emit_signal(SignalMessage &) = 0;
InterfaceAdaptorTable _interfaces;
};
/*
*/
class ObjectProxy;
class InterfaceProxy;
class CallMessage;
typedef std::map<std::string, InterfaceProxy *> InterfaceProxyTable;
class DXXAPI ProxyBase
{
public:
virtual const ObjectProxy *object() const = 0 ;
protected:
InterfaceProxy *find_interface(const std::string &name);
virtual ~ProxyBase()
{}
virtual Message _invoke_method(CallMessage &) = 0;
virtual bool _invoke_method_noreply(CallMessage &call) = 0;
/*!
* \brief Perform a non-blocking method invocation.
* \details Queues a message to send, as with _invoke_method(), but instead
* of blocking to wait for a reply, immediately returns a DBus::PendingCall
* used to receive the reply asynchronously.
*
* The PendingCall is owned by the caller, and must be disposed of using
* _remove_pending_call().
*/
virtual PendingCall *_invoke_method_async(CallMessage &call, int timeout = -1) = 0;
/*!
* \brief Deletes the supplied PendingCall without cancelling it.
* \param pending The PendingCall to be deleted.
*/
virtual void _remove_pending_call(PendingCall *pending) = 0;
InterfaceProxyTable _interfaces;
};
class DXXAPI Interface
{
public:
Interface(const std::string &name);
virtual ~Interface();
inline const std::string &name() const;
private:
std::string _name;
};
/*
*/
const std::string &Interface::name() const
{
return _name;
}
/*
*/
typedef std::map< std::string, Slot<Message, const CallMessage &> > MethodTable;
typedef std::map< std::string, Variant > PropertyDict;
class DXXAPI InterfaceAdaptor : public Interface, public virtual AdaptorBase
{
public:
InterfaceAdaptor(const std::string &name);
Message dispatch_method(const CallMessage &);
void emit_signal(const SignalMessage &);
Variant *get_property(const std::string &name);
void set_property(const std::string &name, Variant &value);
PropertyDict *get_all_properties();
virtual const IntrospectedInterface *introspect() const
{
return NULL;
}
protected:
MethodTable _methods;
PropertyTable _properties;
};
/*
*/
typedef std::map< std::string, Slot<void, const SignalMessage &> > SignalTable;
class DXXAPI InterfaceProxy : public Interface, public virtual ProxyBase
{
public:
InterfaceProxy(const std::string &name);
Message invoke_method(const CallMessage &);
bool invoke_method_noreply(const CallMessage &call);
/*!
* \brief Perform a non-blocking method invocation.
* \details Queues a message to send, as with invoke_method(), but instead
* of blocking to wait for a reply, immediately returns a DBus::PendingCall
* used to receive the reply asynchronously.
*
* The PendingCall is owned by the caller, and must be disposed of using
* remove_pending_call().
*/
PendingCall *invoke_method_async(const CallMessage &call, int timeout = -1);
bool dispatch_signal(const SignalMessage &);
protected:
/*!
* \brief Deletes the supplied PendingCall without cancelling it.
* \param pending The PendingCall to be deleted.
*/
void remove_pending_call(PendingCall *pending);
SignalTable _signals;
};
# define register_method(interface, method, callback) \
InterfaceAdaptor::_methods[ #method ] = \
new ::DBus::Callback< interface, ::DBus::Message, const ::DBus::CallMessage &>(this, & interface :: callback);
# define bind_property(variable, type, can_read, can_write) \
InterfaceAdaptor::_properties[ #variable ].read = can_read; \
InterfaceAdaptor::_properties[ #variable ].write = can_write; \
InterfaceAdaptor::_properties[ #variable ].sig = type; \
variable.bind(InterfaceAdaptor::_properties[ #variable ]);
# define connect_signal(interface, signal, callback) \
InterfaceProxy::_signals[ #signal ] = \
new ::DBus::Callback< interface, void, const ::DBus::SignalMessage &>(this, & interface :: callback);
} /* namespace DBus */
#endif//__DBUSXX_INTERFACE_H