/* chatview.cpp - Chat View Copyright (c) 2002-2004 by Olivier Goffart Copyright (c) 2002-2003 by Martijn Klingens Kopete (c) 2002-2004 by the Kopete developers ************************************************************************* * * * 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 "chatview.h" #include "chatmemberslistwidget.h" #include "chatmessagepart.h" #include "chattexteditpart.h" #include "kopetechatwindow.h" #include "kopetechatsession.h" #include "kopetemetacontact.h" #include "kopetepluginmanager.h" #include "kopeteprefs.h" #include "kopeteprotocol.h" #include "kopeteaccount.h" #include "kopeteglobal.h" #include "kopetecontactlist.h" #include "kopeteviewmanager.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include typedef KGenericFactory ChatWindowPluginFactory; K_EXPORT_COMPONENT_FACTORY( kopete_chatwindow, ChatWindowPluginFactory( "kopete_chatwindow" ) ) ChatWindowPlugin::ChatWindowPlugin(TQObject *parent, const char *name, const TQStringList &) : Kopete::ViewPlugin( ChatWindowPluginFactory::instance(), parent, name ) {} KopeteView* ChatWindowPlugin::createView( Kopete::ChatSession *manager ) { return (KopeteView*)new ChatView(manager,this); } class KopeteChatViewPrivate { public: TQString captionText; TQString statusText; bool isActive; bool sendInProgress; bool visibleMembers; }; ChatView::ChatView( Kopete::ChatSession *mgr, ChatWindowPlugin *parent, const char *name ) : KDockMainWindow( 0L, name, 0L ), KopeteView( mgr, parent ) { d = new KopeteChatViewPrivate; d->isActive = false; d->visibleMembers = false; d->sendInProgress = false; m_mainWindow = 0L; membersDock = 0L; membersStatus = Smart; m_tabState = Normal; //FIXME: don't widgets start off hidden anyway? hide(); //Create the view dock widget (KHTML Part), and set it to no docking (lock it in place) viewDock = createDockWidget(TQString::fromLatin1( "viewDock" ), TQPixmap(), 0L,TQString::fromLatin1("viewDock"), TQString::fromLatin1(" ")); m_messagePart = new ChatMessagePart( mgr, viewDock, "m_messagePart" ); viewDock->setWidget(messagePart()->widget()); viewDock->setDockSite(KDockWidget::DockBottom); viewDock->setEnableDocking(KDockWidget::DockNone); //Create the bottom dock widget, with the edit area, statusbar and send button editDock = createDockWidget( TQString::fromLatin1( "editDock" ), TQPixmap(), 0L, TQString::fromLatin1("editDock"), TQString::fromLatin1(" ") ); m_editPart = new ChatTextEditPart( mgr, editDock, "kopeterichtexteditpart" ); // FIXME: is this used these days? it seems totally unnecessary connect( editPart(), TQT_SIGNAL( toggleToolbar(bool)), this, TQT_SLOT(slotToggleRtfToolbar(bool)) ); connect( editPart(), TQT_SIGNAL( messageSent( Kopete::Message & ) ), this, TQT_SIGNAL( messageSent( Kopete::Message & ) ) ); connect( editPart(), TQT_SIGNAL( canSendChanged( bool ) ), this, TQT_SIGNAL( canSendChanged(bool) ) ); connect( editPart(), TQT_SIGNAL( typing(bool) ), mgr, TQT_SLOT( typing(bool) ) ); //Make the edit area dockable for now editDock->setWidget( editPart()->widget() ); editDock->setDockSite( KDockWidget::DockNone ); editDock->setEnableDocking(KDockWidget::DockBottom); //Set the view as the main widget setMainDockWidget( viewDock ); setView(viewDock); //It is possible to drag and drop on this widget. // I had to disable the acceptDrop in the khtml widget to be able to intercept theses events. setAcceptDrops(true); viewDock->setAcceptDrops(false); m_remoteTypingMap.setAutoDelete( true ); //Manager signals connect( mgr, TQT_SIGNAL( displayNameChanged() ), this, TQT_SLOT( slotChatDisplayNameChanged() ) ); connect( mgr, TQT_SIGNAL( contactAdded(const Kopete::Contact*, bool) ), this, TQT_SLOT( slotContactAdded(const Kopete::Contact*, bool) ) ); connect( mgr, TQT_SIGNAL( contactRemoved(const Kopete::Contact*, const TQString&, Kopete::Message::MessageFormat, bool) ), this, TQT_SLOT( slotContactRemoved(const Kopete::Contact*, const TQString&, Kopete::Message::MessageFormat, bool) ) ); connect( mgr, TQT_SIGNAL( onlineStatusChanged( Kopete::Contact *, const Kopete::OnlineStatus & , const Kopete::OnlineStatus &) ), this, TQT_SLOT( slotContactStatusChanged( Kopete::Contact *, const Kopete::OnlineStatus &, const Kopete::OnlineStatus & ) ) ); connect( mgr, TQT_SIGNAL( remoteTyping( const Kopete::Contact *, bool) ), this, TQT_SLOT( remoteTyping(const Kopete::Contact *, bool) ) ); connect( mgr, TQT_SIGNAL( eventNotification( const TQString& ) ), this, TQT_SLOT( setStatusText( const TQString& ) ) ); //Connections to the manager and the ViewManager that every view should have connect( this, TQT_SIGNAL( closing( KopeteView * ) ), KopeteViewManager::viewManager(), TQT_SLOT( slotViewDestroyed( KopeteView * ) ) ); connect( this, TQT_SIGNAL( activated( KopeteView * ) ), KopeteViewManager::viewManager(), TQT_SLOT( slotViewActivated( KopeteView * ) ) ); connect( this, TQT_SIGNAL( messageSent(Kopete::Message &) ), mgr, TQT_SLOT( sendMessage(Kopete::Message &) ) ); connect( mgr, TQT_SIGNAL( messageSuccess() ), this, TQT_SLOT( messageSentSuccessfully() )); // add contacts slotContactAdded( mgr->myself(), true ); for ( TQPtrListIterator it( mgr->members() ); it.current(); ++it ) slotContactAdded( *it, true ); setFocusProxy( editPart()->widget() ); editPart()->widget()->setFocus(); // init actions KStdAction::copy( TQT_TQOBJECT(this), TQT_SLOT(copy()), actionCollection() ); KStdAction::close( TQT_TQOBJECT(this), TQT_SLOT(closeView()),actionCollection() ); setCaption( m_manager->displayName(), false ); // restore docking positions readOptions(); // maybe show chat members createMembersList(); } ChatView::~ChatView() { emit( closing( static_cast(this) ) ); saveOptions(); delete d; } KTextEdit *ChatView::editWidget() { return editPart()->widget(); } TQWidget *ChatView::mainWidget() { return this; } bool ChatView::canSend() { return editPart()->canSend(); } Kopete::Message ChatView::currentMessage() { return editPart()->contents(); } void ChatView::setCurrentMessage( const Kopete::Message &message ) { editPart()->setContents( message ); } void ChatView::cut() { editPart()->edit()->cut(); } void ChatView::copy() { if ( messagePart()->hasSelection() ) messagePart()->copy(); else editPart()->edit()->copy(); } void ChatView::paste() { editPart()->edit()->paste(); } void ChatView::nickComplete() { return editPart()->complete(); } void ChatView::addText( const TQString &text ) { editPart()->addText( text ); } void ChatView::clear() { messagePart()->clear(); } void ChatView::setBgColor( const TQColor &newColor ) { editPart()->setBgColor( newColor ); } void ChatView::setFont() { editPart()->setFont(); } TQFont ChatView::font() { return editPart()->font(); } void ChatView::setFont( const TQFont &font ) { editPart()->setFont( font ); } void ChatView::setFgColor( const TQColor &newColor ) { editPart()->setFgColor( newColor ); } void ChatView::raise( bool activate ) { // this shouldn't change the focus. When the window is raised when a new message arrives // if i am coding, or talking to someone else, i want to end my sentence before switching to // the other chat. i just want to KNOW and SEE the other chat to switch to it later // (except if activate==true) if ( !m_mainWindow || !m_mainWindow->isActiveWindow() || activate ) makeVisible(); if ( !KWin::windowInfo( m_mainWindow->winId(), NET::WMDesktop ).onAllDesktops() ) if( KopetePrefs::prefs()->trayflashNotifySetCurrentDesktopToChatView() && activate ) KWin::setCurrentDesktop( KWin::windowInfo( m_mainWindow->winId(), NET::WMDesktop ).desktop() ); else KWin::setOnDesktop( m_mainWindow->winId(), KWin::currentDesktop() ); if(m_mainWindow->isMinimized()) { m_mainWindow->showNormal(); } m_mainWindow->raise(); /* Removed Nov 2003 According to Zack, the user double-clicking a contact is not valid reason for a non-pager to grab window focus. While I don't agree with this, and it runs contradictory to every other IM out there, commenting this code out to agree with KWin policy. Redirect any bugs relating to the widnow now not grabbing focus on clicking a contact to KWin. - Jason K */ //Will not activate window if user was typing if ( activate ) KWin::activateWindow( m_mainWindow->winId() ); } void ChatView::makeVisible() { if ( !m_mainWindow ) { m_mainWindow = KopeteChatWindow::window( m_manager ); // if ( root ) // root->repaint( true ); emit windowCreated(); } if ( !m_mainWindow->isVisible() ) { m_mainWindow->show(); // scroll down post show and layout, otherwise the geometry is wrong to scroll to the bottom. m_messagePart->keepScrolledDown(); } m_mainWindow->setActiveView( this ); } bool ChatView::isVisible() { return ( m_mainWindow && m_mainWindow->isVisible() ); } bool ChatView::visibleMembersList() { return d->visibleMembers; } bool ChatView::sendInProgress() { return d->sendInProgress; } bool ChatView::closeView( bool force ) { int response = KMessageBox::Continue; if ( !force ) { if ( m_manager->members().count() > 1 && m_manager->account()->isConnected() ) { TQString shortCaption = d->captionText; shortCaption = KStringHandler::rsqueeze( shortCaption ); response = KMessageBox::warningContinueCancel( this, i18n("You are about to leave the group chat session %1.
" "You will not receive future messages from this conversation.
").arg( shortCaption ), i18n( "Closing Group Chat" ), i18n( "Cl&ose Chat" ), TQString::fromLatin1( "AskCloseGroupChat" ) ); } if ( !unreadMessageFrom.isNull() && ( response == KMessageBox::Continue ) ) { response = KMessageBox::warningContinueCancel( this, i18n("You have received a message from %1 in the last " "second. Are you sure you want to close this chat?").arg( unreadMessageFrom ), i18n( "Unread Message" ), i18n( "Cl&ose Chat" ), TQString::fromLatin1("AskCloseChatRecentMessage" ) ); } if ( d->sendInProgress && ( response == KMessageBox::Continue ) ) { response = KMessageBox::warningContinueCancel( this, i18n( "You have a message send in progress, which will be " "aborted if this chat is closed. Are you sure you want to close this chat?" ), i18n( "Message in Transit" ), i18n( "Cl&ose Chat" ), TQString::fromLatin1( "AskCloseChatMessageInProgress" ) ); } } if( response == KMessageBox::Continue ) { // Remove the widget from the window it's attached to // and schedule it for deletion if( m_mainWindow ) m_mainWindow->detachChatView( this ); deleteLater(); return true; } return false; } void ChatView::updateChatState( KopeteTabState newState ) { if ( newState == Undefined ) newState = m_tabState; else if ( newState != Typing && ( newState != Changed || ( m_tabState != Message && m_tabState != Highlighted ) ) && ( newState != Message || m_tabState != Highlighted ) ) { //if the new state is not a typing state and we don't already have a message or a highlighted message //change the tab state m_tabState = newState; } newState = m_remoteTypingMap.isEmpty() ? m_tabState : Typing ; emit updateChatState( this, newState ); if( newState != Typing ) { setStatusText( i18n( "One other person in the chat", "%n other people in the chat", m_manager->members().count() ) ); } } void ChatView::setMainWindow( KopeteChatWindow* parent ) { m_mainWindow = parent; } void ChatView::createMembersList() { if ( !membersDock ) { //Create the chat members list membersDock = createDockWidget( TQString::fromLatin1( "membersDock" ), TQPixmap(), 0L, TQString::fromLatin1( "membersDock" ), TQString::fromLatin1( " " ) ); m_membersList = new ChatMembersListWidget( m_manager, this, "m_membersList" ); membersDock->setWidget( m_membersList ); Kopete::ContactPtrList members = m_manager->members(); if ( members.first() && members.first()->metaContact() != 0 ) { membersStatus = static_cast ( members.first()->metaContact()->pluginData ( m_manager->protocol(), TQString::fromLatin1( "MembersListPolicy" ) ).toInt() ); } else { membersStatus = Smart; } if( membersStatus == Smart ) d->visibleMembers = ( m_manager->members().count() > 1 ); else d->visibleMembers = ( membersStatus == Visible ); placeMembersList( membersDockPosition ); } } void ChatView::toggleMembersVisibility() { if( membersDock ) { d->visibleMembers = !d->visibleMembers; membersStatus = d->visibleMembers ? Visible : Hidden; placeMembersList( membersDockPosition ); Kopete::ContactPtrList members = m_manager->members(); if ( members.first()->metaContact() ) { members.first()->metaContact()->setPluginData( m_manager->protocol(), TQString::fromLatin1( "MembersListPolicy" ), TQString::number(membersStatus) ); } //refreshView(); } } void ChatView::placeMembersList( KDockWidget::DockPosition dp ) { // kdDebug(14000) << k_funcinfo << "Members list policy " << membersStatus << // ", visible " << d->visibleMembers << endl; if ( d->visibleMembers ) { membersDockPosition = dp; // look up the dock width int dockWidth; TDEGlobal::config()->setGroup( TQString::fromLatin1( "ChatViewDock" ) ); if( membersDockPosition == KDockWidget::DockLeft ) { dockWidth = TDEGlobal::config()->readNumEntry( TQString::fromLatin1( "membersDock,viewDock:sepPos" ), 30); } else { dockWidth = TDEGlobal::config()->readNumEntry( TQString::fromLatin1( "viewDock,membersDock:sepPos" ), 70); } // Make sure it is shown then place it wherever membersDock->setEnableDocking( KDockWidget::DockLeft | KDockWidget::DockRight ); membersDock->manualDock( viewDock, membersDockPosition, dockWidth ); membersDock->show(); membersDock->setEnableDocking( KDockWidget::DockNone ); } else { // Dock it to the desktop then hide it membersDock->undock(); membersDock->hide(); } if( d->isActive ) m_mainWindow->updateMembersActions(); //refreshView(); } void ChatView::remoteTyping( const Kopete::Contact *contact, bool isTyping ) { // Make sure we (re-)add the timer at the end, because the slot will // remove the first timer // And yes, the const_cast is a bit ugly, but it's only used as key // value in this dictionary (no indirections) so it's basically // harmless. Unfortunately there's no TQConstPtrDictionary in TQt... void *key = const_cast( contact ); m_remoteTypingMap.remove( key ); if( isTyping ) { m_remoteTypingMap.insert( key, new TQTimer(this) ); connect( m_remoteTypingMap[ key ], TQT_SIGNAL( timeout() ), TQT_SLOT( slotRemoteTypingTimeout() ) ); m_remoteTypingMap[ key ]->start( 6000, true ); } // Loop through the map, constructing a string of people typing TQStringList typingList; TQPtrDictIterator it( m_remoteTypingMap ); for( ; it.current(); ++it ) { Kopete::Contact *c = static_cast( it.currentKey() ); TQString nick; if( c->metaContact() && c->metaContact() != Kopete::ContactList::self()->myself() ) { nick = c->metaContact()->displayName(); } else { nick = c->nickName(); } typingList.append( nick ); } // Update the status area if( !typingList.isEmpty() ) { if ( typingList.count() == 1 ) setStatusText( i18n( "%1 is typing a message" ).arg( typingList.first() ) ); else { TQString statusTyping = typingList.join( TQString::fromLatin1( ", " ) ); setStatusText( i18n( "%1 is a list of names", "%1 are typing a message" ).arg( statusTyping ) ); } updateChatState( Typing ); } else { updateChatState(); } } void ChatView::setStatusText( const TQString &status ) { d->statusText = status; if ( d->isActive ) m_mainWindow->setStatus( status ); } const TQString& ChatView::statusText() { return d->statusText; } void ChatView::slotChatDisplayNameChanged() { // This fires whenever a contact or MC changes displayName, so only // update the caption if it changed to avoid unneeded updates that // could cause flickering TQString chatName = m_manager->displayName(); if ( chatName != d->captionText ) setCaption( chatName, true ); } void ChatView::slotPropertyChanged( Kopete::Contact*, const TQString &key, const TQVariant& oldValue, const TQVariant &newValue ) { if ( key == Kopete::Global::Properties::self()->nickName().key() ) { TQString newName=newValue.toString(); TQString oldName=oldValue.toString(); if(KopetePrefs::prefs()->showEvents()) if ( oldName != newName && !oldName.isEmpty()) sendInternalMessage( i18n( "%1 is now known as %2" ). arg( oldName, newName ) ); } } void ChatView::slotDisplayNameChanged( const TQString &oldValue, const TQString &newValue ) { if( KopetePrefs::prefs()->showEvents() ) { if( oldValue != newValue ) sendInternalMessage( i18n( "%1 is now known as %2" ). arg( oldValue, newValue ) ); } } void ChatView::slotContactAdded(const Kopete::Contact *contact, bool suppress) { TQString contactName; // Myself metacontact is not a reliable source. if( contact->metaContact() && contact->metaContact() != Kopete::ContactList::self()->myself() ) { contactName = contact->metaContact()->displayName(); } else { contactName = contact->nickName(); } if( contact->metaContact() && contact->metaContact() != Kopete::ContactList::self()->myself() ) { connect( contact->metaContact(), TQT_SIGNAL( displayNameChanged(const TQString&, const TQString&) ), this, TQT_SLOT( slotDisplayNameChanged(const TQString &, const TQString &) ) ); } else { connect( contact, TQT_SIGNAL( propertyChanged( Kopete::Contact *, const TQString &, const TQVariant &, const TQVariant & ) ), this, TQT_SLOT( slotPropertyChanged( Kopete::Contact *, const TQString &, const TQVariant &, const TQVariant & ) ) ) ; } if( !suppress && m_manager->members().count() > 1 ) sendInternalMessage( i18n("%1 has joined the chat.").arg(contactName) ); if( membersStatus == Smart && membersDock ) { bool shouldShowMembers = ( m_manager->members().count() > 1); if( shouldShowMembers != d->visibleMembers ) { d->visibleMembers = shouldShowMembers; placeMembersList( membersDockPosition ); } } updateChatState(); emit updateStatusIcon( this ); } void ChatView::slotContactRemoved( const Kopete::Contact *contact, const TQString &reason, Kopete::Message::MessageFormat format, bool suppressNotification ) { // kdDebug(14000) << k_funcinfo << endl; if ( contact != m_manager->myself() ) { m_remoteTypingMap.remove( const_cast( contact ) ); TQString contactName; if( contact->metaContact() && contact->metaContact() != Kopete::ContactList::self()->myself() ) { contactName = contact->metaContact()->displayName(); } else { contactName = contact->nickName(); } // When the last person leaves, don't disconnect the signals, since we're in a one-to-one chat if ( m_manager->members().count() > 0 ) { if( contact->metaContact() ) { disconnect( contact->metaContact(), TQT_SIGNAL( displayNameChanged(const TQString&, const TQString&) ), this, TQT_SLOT( slotDisplayNameChanged(const TQString&, const TQString&) ) ); } else { disconnect(contact,TQT_SIGNAL(propertyChanged( Kopete::Contact *, const TQString &, const TQVariant &, const TQVariant & )), this, TQT_SLOT( slotPropertyChanged( Kopete::Contact *, const TQString &, const TQVariant &, const TQVariant & ) ) ) ; } } if ( !suppressNotification ) { if ( reason.isEmpty() ) sendInternalMessage( i18n( "%1 has left the chat." ).arg( contactName ), format ) ; else sendInternalMessage( i18n( "%1 has left the chat (%2)." ).arg( contactName, reason ), format); } } updateChatState(); emit updateStatusIcon( this ); } TQString& ChatView::caption() const { return d->captionText; } void ChatView::setCaption( const TQString &text, bool modified ) { // kdDebug(14000) << k_funcinfo << endl; TQString newCaption = text; //Save this caption d->captionText = text; //Turncate if needed newCaption = KStringHandler::rsqueeze( d->captionText, 20 ); //Call the original set caption KDockMainWindow::setCaption( newCaption, false ); emit updateChatTooltip( this, TQString::fromLatin1("%1").arg( d->captionText ) ); emit updateChatLabel( this, newCaption ); //Blink icon if modified and not active if( !d->isActive && modified ) updateChatState( Changed ); else updateChatState(); //Tell the parent we changed our caption emit( captionChanged( d->isActive ) ); } void ChatView::appendMessage(Kopete::Message &message) { remoteTyping( message.from(), false ); messagePart()->appendMessage(message); if( !d->isActive ) { switch ( message.importance() ) { case Kopete::Message::Highlight: updateChatState( Highlighted ); break; case Kopete::Message::Normal: if ( message.direction() == Kopete::Message::Inbound ) { updateChatState( Message ); break; } // if it's an enternal message or a outgoing, fall thought default: updateChatState( Changed ); } } if( message.direction() == Kopete::Message::Inbound ) { if( message.from()->metaContact() && message.from()->metaContact() != Kopete::ContactList::self()->myself() ) { unreadMessageFrom = message.from()->metaContact()->displayName(); } else { unreadMessageFrom = message.from()->nickName(); } TQTimer::singleShot( 1000, this, TQT_SLOT( slotMarkMessageRead() ) ); } else unreadMessageFrom = TQString(); } void ChatView::slotMarkMessageRead() { unreadMessageFrom = TQString(); } void ChatView::slotToggleRtfToolbar( bool enabled ) { emit rtfEnabled( this, enabled ); } void ChatView::slotContactStatusChanged( Kopete::Contact *contact, const Kopete::OnlineStatus &newStatus, const Kopete::OnlineStatus &oldStatus ) { kdDebug(14000) << k_funcinfo << contact << endl; bool inhibitNotification = ( newStatus.status() == Kopete::OnlineStatus::Unknown || oldStatus.status() == Kopete::OnlineStatus::Unknown ); if ( contact && KopetePrefs::prefs()->showEvents() && !inhibitNotification ) { if ( contact->account() && contact == contact->account()->myself() ) { // Separate notification for the 'self' contact if ( newStatus.status() != Kopete::OnlineStatus::Connecting ) sendInternalMessage( i18n( "You are now marked as %1." ).arg( newStatus.description() ) ); } else if ( !contact->account() || !contact->account()->suppressStatusNotification() ) { // Don't send notifications when we just connected ourselves, i.e. when suppressions are still active if ( contact->metaContact() && contact->metaContact() != Kopete::ContactList::self()->myself() ) { sendInternalMessage( i18n( "%2 is now %1." ) .arg( newStatus.description(), contact->metaContact()->displayName() ) ); } else { TQString nick=contact->nickName(); sendInternalMessage( i18n( "%2 is now %1." ) .arg( newStatus.description(), nick ) ); } } } // update the windows caption slotChatDisplayNameChanged(); emit updateStatusIcon( this ); } void ChatView::sendInternalMessage(const TQString &msg, Kopete::Message::MessageFormat format ) { // When closing kopete, some internal message may be sent because some contact are deleted // these contacts can already be deleted Kopete::Message message = Kopete::Message( 0L /*m_manager->myself()*/ , 0L /*m_manager->members()*/, msg, Kopete::Message::Internal, format ); // (in many case, this is useless to set myself as contact) // TODO: set the contact which initiate the internal message, // so we can later show a icon of it (for example, when he join a chat) messagePart()->appendMessage( message ); } void ChatView::sendMessage() { d->sendInProgress = true; editPart()->sendMessage(); } void ChatView::messageSentSuccessfully() { d->sendInProgress = false; emit messageSuccess( this ); } void ChatView::saveOptions() { TDEConfig *config = TDEGlobal::config(); writeDockConfig ( config, TQString::fromLatin1( "ChatViewDock" ) ); config->setGroup( TQString::fromLatin1( "ChatViewDock" ) ); config->writeEntry( TQString::fromLatin1( "membersDockPosition" ), membersDockPosition ); saveChatSettings(); config->sync(); } void ChatView::saveChatSettings() { Kopete::ContactPtrList contacts = msgManager()->members(); if ( contacts.count() == 0 ) return; Kopete::MetaContact* mc = contacts.first()->metaContact(); if ( contacts.count() > 1 ) return; //can't save with more than one person in chatview if ( !mc ) return; TDEConfig* config = TDEGlobal::config(); TQString contactListGroup = TQString::fromLatin1("chatwindow_") + mc->metaContactId(); config->setGroup( contactListGroup ); config->writeEntry( "EnableRichText", editPart()->richTextEnabled() ); config->writeEntry( "EnableAutoSpellCheck", editPart()->autoSpellCheckEnabled() ); config->sync(); } void ChatView::loadChatSettings() { Kopete::ContactPtrList contacts = msgManager()->members(); if ( contacts.count() > 1 ) return; //can't load with more than one other person in the chat //read settings for metacontact TQString contactListGroup = TQString::fromLatin1("chatwindow_") + contacts.first()->metaContact()->metaContactId(); TDEConfig* config = TDEGlobal::config(); config->setGroup( contactListGroup ); bool enableRichText = config->readBoolEntry( "EnableRichText", true ); editPart()->slotSetRichTextEnabled( enableRichText ); emit rtfEnabled( this, editPart()->richTextEnabled() ); bool enableAutoSpell = config->readBoolEntry( "EnableAutoSpellCheck", false ); emit autoSpellCheckEnabled( this, enableAutoSpell ); } void ChatView::readOptions() { TDEConfig *config = TDEGlobal::config(); /** THIS IS BROKEN !!! */ //dockManager->readConfig ( config, TQString::fromLatin1("ChatViewDock") ); //Work-around to restore dock widget positions config->setGroup( TQString::fromLatin1( "ChatViewDock" ) ); membersDockPosition = static_cast( config->readNumEntry( TQString::fromLatin1( "membersDockPosition" ), KDockWidget::DockRight ) ); TQString dockKey = TQString::fromLatin1( "viewDock" ); if ( d->visibleMembers ) { if( membersDockPosition == KDockWidget::DockLeft ) dockKey.prepend( TQString::fromLatin1( "membersDock," ) ); else if( membersDockPosition == KDockWidget::DockRight ) dockKey.append( TQString::fromLatin1( ",membersDock" ) ); } dockKey.append( TQString::fromLatin1( ",editDock:sepPos" ) ); //kdDebug(14000) << k_funcinfo << "reading splitterpos from key: " << dockKey << endl; int splitterPos = config->readNumEntry( dockKey, 70 ); editDock->manualDock( viewDock, KDockWidget::DockBottom, splitterPos ); viewDock->setDockSite( KDockWidget::DockLeft | KDockWidget::DockRight ); editDock->setEnableDocking( KDockWidget::DockNone ); } void ChatView::setActive( bool value ) { d->isActive = value; if ( d->isActive ) { updateChatState( Normal ); emit( activated( static_cast(this) ) ); } } void ChatView::slotRemoteTypingTimeout() { // Remove the topmost timer from the list. Why does TQPtrDict use void* keys and not typed keys? *sigh* if ( !m_remoteTypingMap.isEmpty() ) remoteTyping( reinterpret_cast( TQPtrDictIterator(m_remoteTypingMap).currentKey() ), false ); } void ChatView::dragEnterEvent ( TQDragEnterEvent * event ) { if( event->provides( "kopete/x-contact" ) ) { TQStringList lst=TQStringList::split( TQChar( 0xE000 ) , TQString::fromUtf8(event->encodedData ( "kopete/x-contact" )) ); if(m_manager->mayInvite() && m_manager->protocol()->pluginId() == lst[0] && m_manager->account()->accountId() == lst[1]) { TQString contact=lst[2]; bool found =false; TQPtrList cts=m_manager->members(); for ( TQPtrListIterator it( cts ); it.current(); ++it ) { if(it.current()->contactId() == contact) { found=true; break; } } if(!found && contact != m_manager->myself()->contactId()) event->accept(); } } else if( event->provides( "kopete/x-metacontact" ) ) { TQString metacontactID=TQString::fromUtf8(event->encodedData ( "kopete/x-metacontact" )); Kopete::MetaContact *m=Kopete::ContactList::self()->metaContact(metacontactID); if( m && m_manager->mayInvite()) { TQPtrList cts=m->contacts(); for ( TQPtrListIterator it( cts ); it.current(); ++it ) { Kopete::Contact *c=it.current(); if(c && c->account() == m_manager->account()) { if( c != m_manager->myself() && !m_manager->members().contains(c) && c->isOnline()) event->accept(); } } } } // make sure it doesn't come from the current chat view - then it's an emoticon else if ( event->provides( "text/uri-list" ) && m_manager->members().count() == 1 && ( event->source() != (TQWidget*)m_messagePart->view()->viewport() ) ) { Kopete::ContactPtrList members = m_manager->members(); Kopete::Contact *contact = members.first(); if ( contact && contact->canAcceptFiles() ) event->accept(); } else KDockMainWindow::dragEnterEvent(event); } void ChatView::dropEvent ( TQDropEvent * event ) { if( event->provides( "kopete/x-contact" ) ) { TQStringList lst=TQStringList::split( TQChar( 0xE000 ) , TQString::fromUtf8(event->encodedData ( "kopete/x-contact" )) ); if(m_manager->mayInvite() && m_manager->protocol()->pluginId() == lst[0] && m_manager->account()->accountId() == lst[1]) { TQString contact=lst[2]; bool found =false; TQPtrList cts=m_manager->members(); for ( TQPtrListIterator it( cts ); it.current(); ++it ) { if(it.current()->contactId() == contact) { found=true; break; } } if(!found && contact != m_manager->myself()->contactId()) m_manager->inviteContact(contact); } } else if( event->provides( "kopete/x-metacontact" ) ) { TQString metacontactID=TQString::fromUtf8(event->encodedData ( "kopete/x-metacontact" )); Kopete::MetaContact *m=Kopete::ContactList::self()->metaContact(metacontactID); if(m && m_manager->mayInvite()) { TQPtrList cts=m->contacts(); for ( TQPtrListIterator it( cts ); it.current(); ++it ) { Kopete::Contact *c=it.current(); if(c && c->account() == m_manager->account() && c->isOnline()) { if( c != m_manager->myself() && !m_manager->members().contains(c) ) m_manager->inviteContact(c->contactId()); } } } } else if ( event->provides( "text/uri-list" ) && m_manager->members().count() == 1 ) { Kopete::ContactPtrList members = m_manager->members(); Kopete::Contact *contact = members.first(); if ( !contact || !contact->canAcceptFiles() || !TQUriDrag::canDecode( event ) ) { event->ignore(); return; } KURL::List urlList; KURLDrag::decode( event, urlList ); for ( KURL::List::Iterator it = urlList.begin(); it != urlList.end(); ++it ) { if ( (*it).isLocalFile() ) { //send a file contact->sendFile( *it ); } else { //this is a URL, send the URL in a message addText( (*it).url() ); } } event->acceptAction(); return; } else KDockMainWindow::dropEvent(event); } void ChatView::registerContextMenuHandler( TQObject *target, const char* slot ) { connect( m_messagePart, TQT_SIGNAL( contextMenuEvent( Kopete::Message &, const TQString &, KPopupMenu * ) ), target, slot ); } void ChatView::registerTooltipHandler( TQObject *target, const char* slot ) { connect( m_messagePart, TQT_SIGNAL( tooltipEvent( Kopete::Message &, const TQString &, TQString & ) ), target, slot ); } #include "chatview.moc" // vim: set noet ts=4 sts=4 sw=4: