/*
Copyright (c) 1999 Preston Brown
Copyright (c) 1999, 2000 Matthias Ettrich
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
#ifndef _DCOPCLIENT_H
#define _DCOPCLIENT_H
#include
#include
#include
#include
class DCOPObjectProxy;
class DCOPClientPrivate;
class DCOPClientTransaction;
typedef QValueList QCStringList;
/**
* Inter-process communication and remote procedure calls
* for KDE applications.
*
* This class provides IPC and RPC for KDE applications. Usually you
* will not have to instantiate one yourself because @ref KApplication
* contains a method to return a pointer to a @ref DCOPClient object which
* can be used for your whole application.
*
* Before being able to send or receive any DCOP messages, you will have
* to attach your client object to the DCOP server, and then register
* your application with a specific name. See @ref attach()
* and @ref registerAs() for
* more information.
*
* Data to be sent should be serialized into a @ref QDataStream which was
* initialized with the @ref QByteArray that you actually intend to send
* the data in. An example of how you might do this:
*
*
* QByteArray data;
* QDataStream arg(data, IO_WriteOnly);
* arg << QString("This is text I am serializing");
* client->send("someApp", "someObject", "someFunction", arg);
*
*
* @see KApplication::dcopClient()
* @author Preston Brown , Matthias Ettrich
*/
class DCOPClient : public QObject
{
Q_OBJECT
public:
/**
* Create a new DCOP client, but do not attach to any server. */
DCOPClient();
/**
* Clean up any open connections and dynamic data.
*/
virtual ~DCOPClient();
/**
* Specify the address of a server to use upon attaching.
*
* If no server address is ever specified, attach will try its best to
* find the server anyway.
*/
static void setServerAddress(const QCString &addr);
/**
* Attach to the DCOP server.
*
* If the connection was already attached,
* the connection will be re-established with the current server address.
*
* Naturally, only attached application can use DCOP services.
*
* If a @ref QApplication object exists then client registers itself as
* @ref QApplication->name() + "-" + .
* If no @ref QApplication object exists the client registers itself as
* "anonymous".
*
* If you want to register differently, you should use @ref registerAs()
* instead.
*
* @return @p true if attaching was successful.
*/
bool attach();
/**
* @internal
* Internal function for @ref KUniqueApplication to register the @ref DCOPClient
* with the application in case the application didn't exist at the
* time the @ref DCOPClient was created.
*/
void bindToApp();
/**
* Detach from the DCOP server.
*/
bool detach();
/**
* Query whether or not the client is attached to the server.
*/
bool isAttached() const;
/**
* Register at the DCOP server.
*
* If the application was already registered,
* the registration will be re-done with the new @ref appId.
*
* @p appId is a @p unique application/program id that the server
* will use to associate requests with. If there is already an application
* registered with the same name, the server will add a number to the
* id to unify it. If @p addPID is true, the PID of the current process
* will be added to id.
*
* Registration is necessary if you want to allow other clients to talk
* to you. They can do so using your @p appId as first parameter
* for @ref send() or @ref call(). If you just want to talk to
* other clients, you
* do not need to register at the server. In that case @ref attach() is
* enough.
* It will implicitly register you as "anonymous".
*
* @return The actual @p appId used for the registration or a null string
* if the registration wasn't successful.
*/
QCString registerAs( const QCString &appId, bool addPID = true );
/**
* Query whether or not the client is registered at the server.
*/
bool isRegistered() const;
/**
* Returns the current app id or a null string if the application
* hasn't yet been registered.
*/
QCString appId() const;
/**
* @return The socket over which DCOP is communicating with the server.
*/
int socket() const;
/**
* Temporarily suspend processing of DCOP events.
* This can be usefull if you need to show e.g. a dialog before
* your application is ready to accept DCOP requests. Normally the
* dialog would start an event loop and in this event loop DCOP
* requests will be handled.
*
* Be aware that not responding to DCOP requests may cause other
* programs that want to communicate with your application, to hang.
*/
void suspend();
/**
* Resume the processing of DCOP events.
* See @ref suspend().
*/
void resume();
/**
* Send a data block to the server.
*
* @param remApp The remote application id.
* @param remObj The name of the remote object.
* @param remFun The remote function in the specified object to call.
* @param data The data to provide to the remote function.
* @param fast (KDE 3.0: remove this)
* Tf set to @p true, a "fast" form of IPC will be used.
* Fast connections are not guaranteed to be implemented, but
* if they are they work only on the local machine, not across
* the network. "fast" is only a hint not an order.
*
* @return Whether or not the server was able to accept the send.
*/
bool send(const QCString &remApp, const QCString &remObj,
const QCString &remFun, const QByteArray &data,
bool fast=false);
/**
* This function acts exactly the same as the above, but the data
* parameter can be specified as a @ref QString for convenience.
*/
bool send(const QCString &remApp, const QCString &remObj,
const QCString &remFun, const QString &data,
bool fast=false);
/**
* Perform a synchronous send and receive.
*
* The parameters are the same as for send, with the exception of
* another @ref QByteArray being provided for results to be
* (optionally) returned in.
*
* A call blocks the application until the process receives the
* answer.
*
* If @p useEventLoop is true, a local event loop will be started after
* 1/10th of a second in order to keep the user interface updated
* (by processing paint events and such) until an answer is received.
*
* @see send()
*/
bool call(const QCString &remApp, const QCString &remObj,
const QCString &remFun, const QByteArray &data,
QCString& replyType, QByteArray &replyData,
bool useEventLoop=false, bool fast=false);
/**
* Search for an object which matches a criteria.
*
* @param remApp The remote application id.
* @param remObj The name of the remote object.
* @param remFun The remote function in the specified object to call.
* This function should return a bool and is used as
* criteria.
* @param data The data to provide to the remote function.
* @param foundApp The remote application id that matched the criteria.
* @param foundObj The remote object that matched the criteria.
* @param fast Tf set to true, a "fast" form of IPC will be used.
* Fast connections are not guaranteed to be implemented, but
* if they are they work only on the local machine, not across
* the network. "fast" is only a hint not an order.
*
* findObject calls @p remFun in the applications and objects identified
* by @p remApp and @p remObj until @p remFun returns true. The name of
* the application and object that returned true are returned in
* @p foundApp and @p foundObj respectively.
*
* If @p remFun is empty a default function is called in the object
* which always returns @p true.
*
* @return true is returned when an object was found for which @p remFun
* returned true. If no such object is the function returns false.
*
* A findObject blocks the application until the process receives the
* answer.
*
* If @p useEventLoop is true, a local event loop will be started after
* 1/10th of a second in order to keep the user interface updated
* (by processing paint events and such) until an answer is received.
*
* @see send()
*/
bool findObject(const QCString &remApp, const QCString &remObj,
const QCString &remFun, const QByteArray &data,
QCString &foundApp, QCString &foundObj,
bool useEventLoop=false, bool fast=false);
/**
* Emit @p signal as DCOP signal from object @p object with @p data as
* arguments
*/
void emitDCOPSignal( const QCString &object, const QCString &signal,
const QByteArray &data);
/* For backwards compatibility */
void emitDCOPSignal( const QCString &signal, const QByteArray &data);
/**
* Connect to a DCOP signal
* @param sender the name of the client that emits the signal. When empty
* the signal will be passed from any client.
* @param senderObj the name of the sending object that emits the signal.
* @param signal the name of the signal. The arguments should match with slot.
* @param receiverObj The name of the object to call
* @param slot The name of the slot to call. Its arguments should match with signal.
* @param Volatile If true, the connection will not be reestablished when
* @p sender unregisters and reregisters with DCOP. In this case the @p sender
* must be registered when the connection is made.
* If false, the connection will be reestablished when @p sender reregisters.
* In this case the connection can be made even if @p sender is not registered
* at that time.
*
* @return false if a connection could not be established.
* This will be the case when
* @li @p Volatile is true and @p sender does not exist.
* @li @p signal and @p slot do not have matching arguments.
*/
bool connectDCOPSignal( const QCString &sender, const QCString &senderObj,
const QCString &signal,
const QCString &receiverObj, const QCString &slot,
bool Volatile);
/* For backwards compatibility */
bool connectDCOPSignal( const QCString &sender, const QCString &signal,
const QCString &receiverObj, const QCString &slot,
bool Volatile);
/**
* Disconnect a DCOP signal
* @param sender the name of the client that emits the signal.
* @param senderObj the name of the object that emits the signal.
* If empty all objects will be disconnected.
* @param signal the name of the signal. The arguments should match with slot.
* @param receiverObj The name of the object the signal is connected to.
* If empty all objects will be disconnected.
* @param slot The name of the slot the signal is connected to.
* If empty all slots will be disconnected.
*
* A special case is when both sender & signal are empty. In this
* case all connections related to @p receiverObj in the current client
* are disconnected. (Both connections from as well as to this object!)
*
* @return false if no connection(s) where removed.
*/
bool disconnectDCOPSignal( const QCString &sender, const QCString &senderObj,
const QCString &signal,
const QCString &receiverObj, const QCString &slot);
/* For backwards compatibility */
bool disconnectDCOPSignal( const QCString &sender, const QCString &signal,
const QCString &receiverObj, const QCString &slot);
/**
* Reimplement to handle app-wide function calls unassociated w/an object.
*
* Note that @p fun is normalized. See @ref normalizeFunctionSignature().
*
* If you do not want to reimplement this function for whatever reason,
* you can also use a default object or a @ref DCOPObjectProxy.
*
* @see setDefaultObject()
*/
virtual bool process(const QCString &fun, const QByteArray &data,
QCString& replyType, QByteArray &replyData);
/**
* Delay the reply of the current function call
* until @ref endTransaction() is called.
*
* This allows a server to queue requests.
*
* Note: Should be called from inside @ref process() only!
*/
DCOPClientTransaction *beginTransaction( );
/**
* Send the delayed reply of a function call.
*/
void endTransaction( DCOPClientTransaction *, QCString& replyType, QByteArray &replyData);
/**
* Test whether the current function call is delayed.
*
* Note: Should be called from inside @ref process() only!
* @return The ID of the current transaction or
* 0 if no transaction is going on.
*/
Q_INT32 transactionId();
/**
* Check whether @p remApp is registered with the @ref DCOPServer.
* @return @p true if the remote application is registered, otherwise @p false.
*/
bool isApplicationRegistered( const QCString& remApp);
/**
* Retrieves the list of all currently registered applications.
*/
QCStringList registeredApplications();
/**
* Retrieves the list of objects of the remote application @p remApp
*/
QCStringList remoteObjects( const QCString& remApp, bool *ok = 0 );
/**
* Retrieves the list of interfaces of the remote object @p remObj
* of application @p remApp
*/
QCStringList remoteInterfaces( const QCString& remApp, const QCString& remObj , bool *ok = 0 );
/**
* Retrieves the list of functions of the remote object @p remObj
* of application @p remApp
*/
QCStringList remoteFunctions( const QCString& remApp, const QCString& remObj , bool *ok = 0 );
/**
* Receive a DCOPSend or DCOPCall message from the server.
*
* @param app The application the message was intended for. Should be
* equal to our appId that we passed when the @ref DCOPClient was
* created.
* @param obj The name of the object to pass the data on to.
* @param fun The name of the function in the object to call.
* @param data The arguments for the function.
* @internal
*/
bool receive(const QCString &app, const QCString &obj,
const QCString &fun, const QByteArray& data,
QCString& replyType, QByteArray &replyData);
/**
* Receive a @p DCOPFind message from the server.
*
* @param app The application the message was intended for. Should be
* equal to our appId that we passed when the @ref DCOPClient was
* created.
* @param obj The name of the object to pass the data on to.
* @param fun The name of the function in the object to call.
* @param data The arguments for the function.
* @internal
*/
bool find(const QCString &app, const QCString &obj,
const QCString &fun, const QByteArray& data,
QCString& replyType, QByteArray &replyData);
/**
* Normalize the function signature @p fun.
*
* A normalized signature doesn't contain any unnecessary whitespace
* anymore. The remaining whitespace consists of single blanks only (0x20).
*
* Example for a normalized signature:
*
* "someFunction(QString,int)"
*
*
* When using @ref send() or @ref call(), normalization is done
* automatically for you.
*
*/
static QCString normalizeFunctionSignature( const QCString& fun );
/**
* Retrieve the @p appId of the last application that talked to us.
*/
QCString senderId() const;
/**
* Installs object @p objId as application-wide default object.
*
* All app-wide messages that have not been processed by the dcopclient
* will be send further to @p objId.
*/
void setDefaultObject( const QCString& objId );
/**
* Retrieve the current default object or an empty string if no object is
* installed as default object.
*
* A default object receives application-wide messages that have not
* been processed by the @ref DCOPClient itself.
*/
QCString defaultObject() const;
/**
* Enable / disable the @ref applicationRegistered() /
* @ref applicationRemoved() signals.
* Note that a counter is maintained about how often this method
* was called. If this method is called twice with @p enabled set to
* true, notifications will be enabled until it was called with
* @p enabled set to false as often.
*
* They are disabled by default. */
void setNotifications( bool enabled );
/**
* Tells the dcopserver to treat the client as daemon client, not
* as regular client.
* If the number of regular clients drops down to zero, the
* dcopserver will emit a KDE termination signal after 10
* seconds.
*/
void setDaemonMode( bool daemonMode );
/**
* Returns the application's main dcop client. The main client can
* be used by objects that do not have any specific access to a dcop
* client. In KDE applications, the main client usually is the same
* as KAppliction::dcopClient().
*/
static DCOPClient* mainClient();
/**
* Sets the application's main dcop client. The main client can
* be used by objects that do not have any specific access to a dcop
* client. In KDE applications, the main client usually is the same
* as KAppliction::dcopClient().
*/
static void setMainClient( DCOPClient* );
/**
* @internal Do not use.
*/
static void emergencyClose();
/**
* @internal File with information how to reach the dcopserver.
*/
static QCString dcopServerFile();
signals:
/**
* Indicate that the application @p appId has been registered with
* the server we are attached to.
*
* You need to call @ref setNotifications() first, to tell the @ref DCOPServer
* that you want to get these events
*/
void applicationRegistered( const QCString& appId );
/**
* Indicate that the formerly registered application @p appId has
* been removed.
*
* You need to call @ref setNotifications() first, to tell the @ref
* DCOPServer that you want to get these events
*/
void applicationRemoved( const QCString& appId );
/**
* Indicate that the process of establishing DCOP communications failed
* in some manner.
*
* Usually attached to a dialog box or some other visual
* aid.
*/
void attachFailed(const QString &msg);
/**
* Indicate that user input shall be blocked or released,
* depending on the argument.
*
* The signal is emitted whenever the client has to wait too long
* (i.e. more than 1/10 of a second) for an answer to a
* synchronous call. In that case, it will enter a local event
* loop to keep the GUI updated until finally an answer arrives.
*
* In KDE, the @ref KApplication object connects to this signal to be
* able to block any user input (i.e. mouse and key events) while
* we are waiting for an answer. If we did not do this, the
* application might end up in an illegal state, as a keyboard
* shortcut or a mouse action might cause another dcop call to be
* issued.
*/
void blockUserInput( bool );
public slots:
protected slots:
void processSocketData(int socknum);
private slots:
void processPostedMessagesInternal();
protected:
private:
DCOPClientPrivate *d;
bool attachInternal( bool registerAsAnonymous = TRUE );
bool callInternal(const QCString &remApp, const QCString &remObj,
const QCString &remFun, const QByteArray &data,
QCString& replyType, QByteArray &replyData,
bool useEventLoop, bool fast, int minor_opcode);
};
#endif
Generated by: dfaure on kde.faure.org on Thu Jan 17 22:14:29 2002, using kdoc 2.0a53. |