|
|
|
/*
|
|
|
|
kopeteviewmanager.cpp - View Manager
|
|
|
|
|
|
|
|
Copyright (c) 2003 by Jason Keirstead <jason@keirstead.org>
|
|
|
|
Kopete (c) 2002-2005 by the Kopete developers <kopete-devel@kde.org>
|
|
|
|
|
|
|
|
*************************************************************************
|
|
|
|
* *
|
|
|
|
* This program is free software; you can redistribute it and/or modify *
|
|
|
|
* it under the terms of the GNU General Public License as published by *
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or *
|
|
|
|
* (at your option) any later version. *
|
|
|
|
* *
|
|
|
|
*************************************************************************
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <tdeapplication.h>
|
|
|
|
#include <tdeconfig.h>
|
|
|
|
#include <kdebug.h>
|
|
|
|
#include <tdelocale.h>
|
|
|
|
#include <tqptrlist.h>
|
|
|
|
#include <tqstylesheet.h>
|
|
|
|
#include <kplugininfo.h>
|
|
|
|
#include <knotification.h>
|
|
|
|
#include <tdeglobal.h>
|
|
|
|
#include <twin.h>
|
|
|
|
|
|
|
|
#include "kopeteprefs.h"
|
|
|
|
#include "kopeteaccount.h"
|
|
|
|
#include "kopetepluginmanager.h"
|
|
|
|
#include "kopeteviewplugin.h"
|
|
|
|
#include "kopetechatsessionmanager.h"
|
|
|
|
#include "kopetemetacontact.h"
|
|
|
|
#include "kopetenotifyevent.h"
|
|
|
|
#include "kopetemessageevent.h"
|
|
|
|
#include "kopeteview.h"
|
|
|
|
//#include "systemtray.h"
|
|
|
|
|
|
|
|
#include "kopeteviewmanager.h"
|
|
|
|
|
|
|
|
typedef TQMap<Kopete::ChatSession*,KopeteView*> ManagerMap;
|
|
|
|
typedef TQPtrList<Kopete::MessageEvent> EventList;
|
|
|
|
|
|
|
|
struct KopeteViewManagerPrivate
|
|
|
|
{
|
|
|
|
ManagerMap managerMap;
|
|
|
|
EventList eventList;
|
|
|
|
KopeteView *activeView;
|
|
|
|
|
|
|
|
bool useQueueOrStack;
|
|
|
|
bool raiseWindow;
|
|
|
|
bool queueUnreadMessages;
|
|
|
|
bool queueOnlyHighlightedMessagesInGroupChats;
|
|
|
|
bool queueOnlyMessagesOnAnotherDesktop;
|
|
|
|
bool balloonNotifyIgnoreClosesChatView;
|
|
|
|
bool foreignMessage;
|
|
|
|
};
|
|
|
|
|
|
|
|
KopeteViewManager *KopeteViewManager::s_viewManager = 0L;
|
|
|
|
|
|
|
|
KopeteViewManager *KopeteViewManager::viewManager()
|
|
|
|
{
|
|
|
|
if( !s_viewManager )
|
|
|
|
s_viewManager = new KopeteViewManager();
|
|
|
|
return s_viewManager;
|
|
|
|
}
|
|
|
|
|
|
|
|
KopeteViewManager::KopeteViewManager()
|
|
|
|
{
|
|
|
|
s_viewManager=this;
|
|
|
|
d = new KopeteViewManagerPrivate;
|
|
|
|
d->activeView = 0L;
|
|
|
|
d->foreignMessage=false;
|
|
|
|
|
|
|
|
connect( KopetePrefs::prefs(), TQ_SIGNAL( saved() ), this, TQ_SLOT( slotPrefsChanged() ) );
|
|
|
|
|
|
|
|
connect( Kopete::ChatSessionManager::self() , TQ_SIGNAL( display( Kopete::Message &, Kopete::ChatSession *) ),
|
|
|
|
this, TQ_SLOT ( messageAppended( Kopete::Message &, Kopete::ChatSession *) ) );
|
|
|
|
|
|
|
|
connect( Kopete::ChatSessionManager::self() , TQ_SIGNAL( readMessage() ),
|
|
|
|
this, TQ_SLOT ( nextEvent() ) );
|
|
|
|
|
|
|
|
slotPrefsChanged();
|
|
|
|
}
|
|
|
|
|
|
|
|
KopeteViewManager::~KopeteViewManager()
|
|
|
|
{
|
|
|
|
// kdDebug(14000) << k_funcinfo << endl;
|
|
|
|
|
|
|
|
//delete all open chatwindow.
|
|
|
|
ManagerMap::Iterator it;
|
|
|
|
for ( it = d->managerMap.begin(); it != d->managerMap.end(); ++it )
|
|
|
|
it.data()->closeView( true ); //this does not clean the map, but we don't care
|
|
|
|
|
|
|
|
delete d;
|
|
|
|
}
|
|
|
|
|
|
|
|
void KopeteViewManager::slotPrefsChanged()
|
|
|
|
{
|
|
|
|
d->useQueueOrStack = KopetePrefs::prefs()->useQueue() || KopetePrefs::prefs()->useStack();
|
|
|
|
d->raiseWindow = KopetePrefs::prefs()->raiseMsgWindow();
|
|
|
|
d->queueUnreadMessages = KopetePrefs::prefs()->queueUnreadMessages();
|
|
|
|
d->queueOnlyHighlightedMessagesInGroupChats = KopetePrefs::prefs()->queueOnlyHighlightedMessagesInGroupChats();
|
|
|
|
d->queueOnlyMessagesOnAnotherDesktop = KopetePrefs::prefs()->queueOnlyMessagesOnAnotherDesktop();
|
|
|
|
d->balloonNotifyIgnoreClosesChatView = KopetePrefs::prefs()->balloonNotifyIgnoreClosesChatView();
|
|
|
|
}
|
|
|
|
|
|
|
|
KopeteView *KopeteViewManager::view( Kopete::ChatSession* session, const TQString &requestedPlugin )
|
|
|
|
{
|
|
|
|
// kdDebug(14000) << k_funcinfo << endl;
|
|
|
|
|
|
|
|
if( d->managerMap.contains( session ) && d->managerMap[ session ] )
|
|
|
|
{
|
|
|
|
return d->managerMap[ session ];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Kopete::PluginManager *pluginManager = Kopete::PluginManager::self();
|
|
|
|
Kopete::ViewPlugin *viewPlugin = 0L;
|
|
|
|
|
|
|
|
TQString pluginName = requestedPlugin.isEmpty() ? KopetePrefs::prefs()->interfacePreference() : requestedPlugin;
|
|
|
|
if( !pluginName.isEmpty() )
|
|
|
|
{
|
|
|
|
viewPlugin = (Kopete::ViewPlugin*)pluginManager->loadPlugin( pluginName );
|
|
|
|
|
|
|
|
if( !viewPlugin )
|
|
|
|
{
|
|
|
|
kdWarning(14000) << "Requested view plugin, " << pluginName <<
|
|
|
|
", was not found. Falling back to chat window plugin" << endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if( !viewPlugin )
|
|
|
|
viewPlugin = (Kopete::ViewPlugin*)pluginManager->loadPlugin( TQString::fromLatin1("kopete_chatwindow") );
|
|
|
|
|
|
|
|
if( viewPlugin )
|
|
|
|
{
|
|
|
|
KopeteView *newView = viewPlugin->createView(session);
|
|
|
|
|
|
|
|
d->foreignMessage = false;
|
|
|
|
d->managerMap.insert( session, newView );
|
|
|
|
|
|
|
|
connect( session, TQ_SIGNAL( closing(Kopete::ChatSession *) ),
|
|
|
|
this, TQ_SLOT(slotChatSessionDestroyed(Kopete::ChatSession*)) );
|
|
|
|
|
|
|
|
return newView;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
kdError(14000) << "Could not create a view, no plugins available!" << endl;
|
|
|
|
return 0L;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void KopeteViewManager::messageAppended( Kopete::Message &msg, Kopete::ChatSession *manager)
|
|
|
|
{
|
|
|
|
// kdDebug(14000) << k_funcinfo << endl;
|
|
|
|
|
|
|
|
bool outgoingMessage = ( msg.direction() == Kopete::Message::Outbound );
|
|
|
|
|
|
|
|
if( !outgoingMessage || d->managerMap.contains( manager ) )
|
|
|
|
{
|
|
|
|
d->foreignMessage=!outgoingMessage; //let know for the view we are about to create
|
|
|
|
manager->view(true,msg.requestedPlugin())->appendMessage( msg );
|
|
|
|
d->foreignMessage=false; //the view is created, reset the flag
|
|
|
|
|
|
|
|
bool appendMessageEvent = d->useQueueOrStack;
|
|
|
|
|
|
|
|
TQWidget *w;
|
|
|
|
if( d->queueUnreadMessages && ( w = dynamic_cast<TQWidget*>(view( manager )) ) )
|
|
|
|
{
|
|
|
|
// append msg event to queue if chat window is active but not the chat view in it...
|
|
|
|
appendMessageEvent = appendMessageEvent && !(w->isActiveWindow() && manager->view() == d->activeView);
|
|
|
|
// ...and chat window is on another desktop
|
|
|
|
appendMessageEvent = appendMessageEvent && (!d->queueOnlyMessagesOnAnotherDesktop || !KWin::windowInfo( w->topLevelWidget()->winId(), NET::WMDesktop ).isOnCurrentDesktop());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// append if no chat window exists already
|
|
|
|
appendMessageEvent = appendMessageEvent && !view( manager )->isVisible();
|
|
|
|
}
|
|
|
|
|
|
|
|
// in group chats always append highlighted messages to queue
|
|
|
|
appendMessageEvent = appendMessageEvent && (!d->queueOnlyHighlightedMessagesInGroupChats || manager->members().count() == 1 || msg.importance() == Kopete::Message::Highlight);
|
|
|
|
|
|
|
|
if( appendMessageEvent )
|
|
|
|
{
|
|
|
|
if ( !outgoingMessage )
|
|
|
|
{
|
|
|
|
Kopete::MessageEvent *event=new Kopete::MessageEvent(msg,manager);
|
|
|
|
d->eventList.append( event );
|
|
|
|
connect(event, TQ_SIGNAL(done(Kopete::MessageEvent *)), this, TQ_SLOT(slotEventDeleted(Kopete::MessageEvent *)));
|
|
|
|
Kopete::ChatSessionManager::self()->postNewEvent(event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if( d->eventList.isEmpty() )
|
|
|
|
{
|
|
|
|
readMessages( manager, outgoingMessage );
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( !outgoingMessage && ( !manager->account()->isAway() || KopetePrefs::prefs()->soundIfAway() )
|
|
|
|
&& msg.direction() != Kopete::Message::Internal )
|
|
|
|
{
|
|
|
|
TQWidget *w=dynamic_cast<TQWidget*>(manager->view(false));
|
|
|
|
TDEConfig *config = TDEGlobal::config();
|
|
|
|
config->setGroup("General");
|
|
|
|
if( (!manager->view(false) || !w || manager->view() != d->activeView ||
|
|
|
|
config->readBoolEntry("EventIfActive", true) || !w->isActiveWindow())
|
|
|
|
&& msg.from())
|
|
|
|
{
|
|
|
|
TQString msgFrom = TQString();
|
|
|
|
if( msg.from()->metaContact() )
|
|
|
|
msgFrom = msg.from()->metaContact()->displayName();
|
|
|
|
else
|
|
|
|
msgFrom = msg.from()->contactId();
|
|
|
|
|
|
|
|
TQString msgText = msg.plainBody();
|
|
|
|
if( msgText.length() > 90 )
|
|
|
|
msgText = msgText.left(88) + TQString::fromLatin1("...");
|
|
|
|
|
|
|
|
TQString event;
|
|
|
|
TQString body =i18n( "<qt>Incoming message from %1<br>\"%2\"</qt>" );
|
|
|
|
|
|
|
|
switch( msg.importance() )
|
|
|
|
{
|
|
|
|
case Kopete::Message::Low:
|
|
|
|
event = TQString::fromLatin1( "kopete_contact_lowpriority" );
|
|
|
|
break;
|
|
|
|
case Kopete::Message::Highlight:
|
|
|
|
event = TQString::fromLatin1( "kopete_contact_highlight" );
|
|
|
|
body = i18n( "<qt>A highlighted message arrived from %1<br>\"%2\"</qt>" );
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
event = TQString::fromLatin1( "kopete_contact_incoming" );
|
|
|
|
}
|
|
|
|
KNotification *notify=KNotification::event(msg.from()->metaContact() , event, body.arg( TQStyleSheet::escape(msgFrom), TQStyleSheet::escape(msgText) ), 0, /*msg.from()->metaContact(),*/
|
|
|
|
w , i18n("View") );
|
|
|
|
|
|
|
|
connect(notify,TQ_SIGNAL(activated(unsigned int )), manager , TQ_SLOT(raiseView()) );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void KopeteViewManager::readMessages( Kopete::ChatSession *manager, bool outgoingMessage, bool activate )
|
|
|
|
{
|
|
|
|
// kdDebug( 14000 ) << k_funcinfo << endl;
|
|
|
|
d->foreignMessage=!outgoingMessage; //let know for the view we are about to create
|
|
|
|
KopeteView *thisView = manager->view( true );
|
|
|
|
d->foreignMessage=false; //the view is created, reset the flag
|
|
|
|
if( ( outgoingMessage && !thisView->isVisible() ) || d->raiseWindow || activate )
|
|
|
|
thisView->raise( activate );
|
|
|
|
else if( !thisView->isVisible() )
|
|
|
|
thisView->makeVisible();
|
|
|
|
|
|
|
|
TQPtrListIterator<Kopete::MessageEvent> it( d->eventList );
|
|
|
|
Kopete::MessageEvent* event;
|
|
|
|
while ( ( event = it.current() ) != 0 )
|
|
|
|
{
|
|
|
|
++it;
|
|
|
|
if ( event->message().manager() == manager )
|
|
|
|
{
|
|
|
|
event->apply();
|
|
|
|
d->eventList.remove( event );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void KopeteViewManager::slotEventDeleted( Kopete::MessageEvent *event )
|
|
|
|
{
|
|
|
|
// kdDebug(14000) << k_funcinfo << endl;
|
|
|
|
Kopete::ChatSession *kmm=event->message().manager();
|
|
|
|
if(!kmm)
|
|
|
|
return;
|
|
|
|
|
|
|
|
d->eventList.remove( event );
|
|
|
|
|
|
|
|
if ( event->state() == Kopete::MessageEvent::Applied )
|
|
|
|
{
|
|
|
|
readMessages( kmm, false, true );
|
|
|
|
}
|
|
|
|
else if ( event->state() == Kopete::MessageEvent::Ignored && d->balloonNotifyIgnoreClosesChatView )
|
|
|
|
{
|
|
|
|
bool bAnotherWithThisManager = false;
|
|
|
|
for( TQPtrListIterator<Kopete::MessageEvent> it( d->eventList ); it; ++it )
|
|
|
|
{
|
|
|
|
Kopete::MessageEvent *event = it.current();
|
|
|
|
if ( event->message().manager() == kmm )
|
|
|
|
bAnotherWithThisManager = true;
|
|
|
|
}
|
|
|
|
if ( !bAnotherWithThisManager && kmm->view( false ) )
|
|
|
|
kmm->view()->closeView( true );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void KopeteViewManager::nextEvent()
|
|
|
|
{
|
|
|
|
// kdDebug( 14000 ) << k_funcinfo << endl;
|
|
|
|
|
|
|
|
if( d->eventList.isEmpty() )
|
|
|
|
return;
|
|
|
|
|
|
|
|
Kopete::MessageEvent* event = d->eventList.first();
|
|
|
|
|
|
|
|
if ( event )
|
|
|
|
event->apply();
|
|
|
|
}
|
|
|
|
|
|
|
|
void KopeteViewManager::slotViewActivated( KopeteView *view )
|
|
|
|
{
|
|
|
|
// kdDebug( 14000 ) << k_funcinfo << endl;
|
|
|
|
d->activeView = view;
|
|
|
|
|
|
|
|
TQPtrListIterator<Kopete::MessageEvent> it ( d->eventList );
|
|
|
|
Kopete::MessageEvent* event;
|
|
|
|
while ( ( event = it.current() ) != 0 )
|
|
|
|
{
|
|
|
|
++it;
|
|
|
|
if ( event->message().manager() == view->msgManager() )
|
|
|
|
event->deleteLater();
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void KopeteViewManager::slotViewDestroyed( KopeteView *closingView )
|
|
|
|
{
|
|
|
|
// kdDebug( 14000 ) << k_funcinfo << endl;
|
|
|
|
|
|
|
|
if( d->managerMap.contains( closingView->msgManager() ) )
|
|
|
|
{
|
|
|
|
d->managerMap.remove( closingView->msgManager() );
|
|
|
|
// closingView->msgManager()->setCanBeDeleted( true );
|
|
|
|
}
|
|
|
|
|
|
|
|
if( closingView == d->activeView )
|
|
|
|
d->activeView = 0L;
|
|
|
|
}
|
|
|
|
|
|
|
|
void KopeteViewManager::slotChatSessionDestroyed( Kopete::ChatSession *manager )
|
|
|
|
{
|
|
|
|
// kdDebug( 14000 ) << k_funcinfo << endl;
|
|
|
|
|
|
|
|
if( d->managerMap.contains( manager ) )
|
|
|
|
{
|
|
|
|
KopeteView *v=d->managerMap[ manager ];
|
|
|
|
v->closeView( true );
|
|
|
|
delete v; //closeView call deleteLater, but in this case this is not enough, because some signal are called that case crash
|
|
|
|
d->managerMap.remove( manager );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
KopeteView* KopeteViewManager::activeView() const
|
|
|
|
{
|
|
|
|
return d->activeView;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#include "kopeteviewmanager.moc"
|