You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
konversation/konversation/src/channel.cpp

2961 lines
96 KiB

/*
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.
*/
/*
Copyright (C) 2002 Dario Abatianni <eisfuchs@tigress.com>
Copyright (C) 2004-2006 Peter Simonsson <psn@linux.se>
Copyright (C) 2006-2008 Eike Hein <hein@kde.org>
*/
#include "channel.h"
#include "konversationapplication.h"
#include "server.h"
#include "blowfish.h"
#include "nick.h"
#include "nicklistview.h"
#include "quickbutton.h"
#include "modebutton.h"
#include "ircinput.h"
#include "ircviewbox.h"
#include "ircview.h"
#include <kabc/addressbook.h>
#include <kabc/stdaddressbook.h>
#include "common.h"
#include "topiclabel.h"
#include "channeloptionsdialog.h"
#include "notificationhandler.h"
#include "viewcontainer.h"
#include "linkaddressbook/linkaddressbookui.h"
#include "linkaddressbook/addressbook.h"
#include <tqlabel.h>
#include <tqvbox.h>
#include <tqevent.h>
#include <tqhbox.h>
#include <tqgrid.h>
#include <tqdragobject.h>
#include <tqsizepolicy.h>
#include <tqheader.h>
#include <tqregexp.h>
#include <tqtooltip.h>
#include <tqsplitter.h>
#include <tqcheckbox.h>
#include <tqtimer.h>
#include <tqcombobox.h>
#include <tqtextcodec.h>
#include <tqwhatsthis.h>
#include <tqtoolbutton.h>
#include <tqlayout.h>
#include <kprocess.h>
#include <klineedit.h>
#include <kinputdialog.h>
#include <kpassdlg.h>
#include <klocale.h>
#include <kdebug.h>
#include <kglobalsettings.h>
#include <tdeversion.h>
#include <kmessagebox.h>
#include <kiconloader.h>
#include <twin.h>
Channel::Channel(TQWidget* parent, TQString _name) : ChatWindow(parent)
{
// init variables
//HACK I needed the channel name at time of setServer, but setName needs m_server..
// This effectively assigns the name twice, but none of the other logic has been moved or updated.
name=_name;
m_processingTimer = 0;
m_delayedSortTimer = 0;
m_optionsDialog = NULL;
m_pendingChannelNickLists.clear();
m_currentIndex = 0;
m_opsToAdd = 0;
nicks = 0;
ops = 0;
completionPosition = 0;
nickChangeDialog = 0;
channelCommand = false;
m_joined = false;
quickButtonsChanged = false;
quickButtonsState = false;
modeButtonsChanged = false;
modeButtonsState = false;
awayChanged = false;
awayState = false;
splittersInitialized = false;
topicSplitterHidden = false;
channelSplitterHidden = false;
// flag for first seen topic
topicAuthorUnknown = true;
setType(ChatWindow::Channel);
setChannelEncodingSupported(true);
// Build some size policies for the widgets
TQSizePolicy hfixed = TQSizePolicy(TQSizePolicy::Fixed, TQSizePolicy::Preferred);
TQSizePolicy hmodest = TQSizePolicy(TQSizePolicy::Preferred, TQSizePolicy::Expanding);
TQSizePolicy vmodest = TQSizePolicy(TQSizePolicy::Expanding, TQSizePolicy::Preferred);
TQSizePolicy vfixed = TQSizePolicy(TQSizePolicy::Preferred, TQSizePolicy::Fixed);
TQSizePolicy modest = TQSizePolicy(TQSizePolicy::Preferred, TQSizePolicy::Preferred);
TQSizePolicy greedy = TQSizePolicy(TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding);
m_vertSplitter = new TQSplitter(Qt::Vertical, this);
m_vertSplitter->setOpaqueResize(TDEGlobalSettings::opaqueResize());
TQWidget* topicWidget = new TQWidget(m_vertSplitter);
m_vertSplitter->setResizeMode(topicWidget,TQSplitter::KeepSize);
TQGridLayout* topicLayout = new TQGridLayout(topicWidget, 2, 3, 0, 0);
m_topicButton = new TQToolButton(topicWidget);
m_topicButton->setIconSet(SmallIconSet("edit", 16));
TQToolTip::add(m_topicButton, i18n("Edit Channel Settings"));
connect(m_topicButton, TQT_SIGNAL(clicked()), this, TQT_SLOT(showOptionsDialog()));
topicLine = new Konversation::TopicLabel(topicWidget);
TQWhatsThis::add(topicLine, i18n("<qt>Every channel on IRC has a topic associated with it. This is simply a message that everybody can see.<p>If you are an operator, or the channel mode <em>'T'</em> has not been set, then you can change the topic by clicking the Edit Channel Properties button to the left of the topic. You can also view the history of topics there.</qt>"));
connect(topicLine, TQT_SIGNAL(setStatusBarTempText(const TQString&)), this, TQT_SIGNAL(setStatusBarTempText(const TQString&)));
connect(topicLine, TQT_SIGNAL(clearStatusBarTempText()), this, TQT_SIGNAL(clearStatusBarTempText()));
connect(topicLine,TQT_SIGNAL(popupCommand(int)),this,TQT_SLOT(popupChannelCommand(int)));
topicLayout->addWidget(m_topicButton, 0, 0);
topicLayout->addMultiCellWidget(topicLine, 0, 1, 1, 1);
// The box holding the channel modes
modeBox = new TQHBox(topicWidget);
modeBox->setSizePolicy(hfixed);
modeT = new ModeButton("T",modeBox,0);
modeN = new ModeButton("N",modeBox,1);
modeS = new ModeButton("S",modeBox,2);
modeI = new ModeButton("I",modeBox,3);
modeP = new ModeButton("P",modeBox,4);
modeM = new ModeButton("M",modeBox,5);
modeK = new ModeButton("K",modeBox,6);
modeL = new ModeButton("L",modeBox,7);
TQWhatsThis::add(modeT, i18n("<qt>These control the <em>mode</em> of the channel. Only an operator can change these.<p>The <b>T</b>opic mode means that only the channel operator can change the topic for the channel.</qt>"));
TQWhatsThis::add(modeN, i18n("<qt>These control the <em>mode</em> of the channel. Only an operator can change these.<p><b>N</b>o messages from outside means that users that are not in the channel cannot send messages that everybody in the channel can see. Almost all channels have this set to prevent nuisance messages.</qt>"));
TQWhatsThis::add(modeS, i18n("<qt>These control the <em>mode</em> of the channel. Only an operator can change these.<p>A <b>S</b>ecret channel will not show up in the channel list, nor will any user be able to see that you are in the channel with the <em>WHOIS</em> command or anything similar. Only the people that are in the same channel will know that you are in this channel, if this mode is set.</qt>"));
TQWhatsThis::add(modeI, i18n("<qt>These control the <em>mode</em> of the channel. Only an operator can change these.<p>An <b>I</b>nvite only channel means that people can only join the channel if they are invited. To invite someone, a channel operator needs to issue the command <em>/invite nick</em> from within the channel.</qt>"));
TQWhatsThis::add(modeP, i18n("<qt>These control the <em>mode</em> of the channel. Only an operator can change these.<p>A <b>P</b>rivate channel is shown in a listing of all channels, but the topic is not shown. A user's <em>WHOIS</e> may or may not show them as being in a private channel depending on the IRC server.</qt>"));
TQWhatsThis::add(modeM, i18n("<qt>These control the <em>mode</em> of the channel. Only an operator can change these.<p>A <b>M</b>oderated channel is one where only operators, half-operators and those with voice can talk.</qt>"));
TQWhatsThis::add(modeK, i18n("<qt>These control the <em>mode</em> of the channel. Only an operator can change these.<p>A <b>P</b>rotected channel requires users to enter a password in order to join.</qt>"));
TQWhatsThis::add(modeL, i18n("<qt>These control the <em>mode</em> of the channel. Only an operator can change these.<p>A channel that has a user <b>L</b>imit means that only that many users can be in the channel at any one time. Some channels have a bot that sits in the channel and changes this automatically depending on how busy the channel is.</qt>"));
connect(modeT,TQT_SIGNAL(clicked(int,bool)),this,TQT_SLOT(modeButtonClicked(int,bool)));
connect(modeN,TQT_SIGNAL(clicked(int,bool)),this,TQT_SLOT(modeButtonClicked(int,bool)));
connect(modeS,TQT_SIGNAL(clicked(int,bool)),this,TQT_SLOT(modeButtonClicked(int,bool)));
connect(modeI,TQT_SIGNAL(clicked(int,bool)),this,TQT_SLOT(modeButtonClicked(int,bool)));
connect(modeP,TQT_SIGNAL(clicked(int,bool)),this,TQT_SLOT(modeButtonClicked(int,bool)));
connect(modeM,TQT_SIGNAL(clicked(int,bool)),this,TQT_SLOT(modeButtonClicked(int,bool)));
connect(modeK,TQT_SIGNAL(clicked(int,bool)),this,TQT_SLOT(modeButtonClicked(int,bool)));
connect(modeL,TQT_SIGNAL(clicked(int,bool)),this,TQT_SLOT(modeButtonClicked(int,bool)));
limit=new KLineEdit(modeBox);
TQToolTip::add(limit, i18n("Maximum users allowed in channel"));
TQWhatsThis::add(limit, i18n("<qt>This is the channel user limit - the maximum number of users that can be in the channel at a time. If you are an operator, you can set this. The channel mode <b>T</b>opic (button to left) will automatically be set if set this.</qt>"));
connect(limit,TQT_SIGNAL (returnPressed()),this,TQT_SLOT (channelLimitChanged()) );
connect(limit,TQT_SIGNAL (lostFocus()), this, TQT_SLOT(channelLimitChanged()) );
topicLayout->addWidget(modeBox, 0, 2);
topicLayout->setRowStretch(1, 10);
topicLayout->setColStretch(1, 10);
showTopic(Preferences::showTopic());
showModeButtons(Preferences::showModeButtons());
// (this) The main Box, holding the channel view/topic and the input line
m_horizSplitter = new TQSplitter(m_vertSplitter);
m_horizSplitter->setOpaqueResize( TDEGlobalSettings::opaqueResize() );
// Server will be set later in setServer()
IRCViewBox* ircViewBox = new IRCViewBox(m_horizSplitter, NULL);
setTextView(ircViewBox->ircView());
connect(textView,TQT_SIGNAL(popupCommand(int)),this,TQT_SLOT(popupChannelCommand(int)));
connect(topicLine, TQT_SIGNAL(currentChannelChanged(const TQString&)),textView,TQT_SLOT(setCurrentChannel(const TQString&)));
// The box that holds the Nick List and the quick action buttons
nickListButtons = new TQVBox(m_horizSplitter);
m_horizSplitter->setResizeMode(nickListButtons,TQSplitter::KeepSize);
nickListButtons->setSpacing(spacing());
nicknameListView=new NickListView(nickListButtons, this);
nicknameListView->setHScrollBarMode(TQScrollView::AlwaysOff);
nicknameListView->setSelectionModeExt(TDEListView::Extended);
nicknameListView->setAllColumnsShowFocus(true);
nicknameListView->setSorting(1,true);
nicknameListView->addColumn(TQString());
nicknameListView->addColumn(TQString());
nicknameListView->setColumnWidthMode(1,TDEListView::Maximum);
nicknameListView->header()->hide();
// setResizeMode must be called after all the columns are added
nicknameListView->setResizeMode(TDEListView::LastColumn);
// separate LED from Text a little more
nicknameListView->setColumnWidth(0, 10);
nicknameListView->setColumnAlignment(0, TQt::AlignHCenter);
nicknameListView->installEventFilter(this);
// initialize buttons grid, will be set up in updateQuickButtons
buttonsGrid=0;
// The box holding the Nickname button and Channel input
commandLineBox = new TQHBox(this);
commandLineBox->setSpacing(spacing());
nicknameCombobox = new TQComboBox(commandLineBox);
nicknameCombobox->setEditable(true);
nicknameCombobox->insertStringList(Preferences::nicknameList());
TQWhatsThis::add(nicknameCombobox, i18n("<qt>This shows your current nick, and any alternatives you have set up. If you select or type in a different nickname, then a request will be sent to the IRC server to change your nick. If the server allows it, the new nickname will be selected. If you type in a new nickname, you need to press 'Enter' at the end.<p>You can add change the alternative nicknames from the <em>Identities</em> option in the <em>File</em> menu.</qt>"));
oldNick = nicknameCombobox->currentText();
awayLabel = new TQLabel(i18n("(away)"), commandLineBox);
awayLabel->hide();
blowfishLabel = new TQLabel(commandLineBox);
blowfishLabel->hide();
blowfishLabel->setPixmap(TDEGlobal::iconLoader()->loadIcon("encrypted", KIcon::Toolbar));
channelInput = new IRCInput(commandLineBox);
getTextView()->installEventFilter(channelInput);
topicLine->installEventFilter(channelInput);
channelInput->installEventFilter(this);
// Set the widgets size policies
m_topicButton->setSizePolicy(TQSizePolicy(TQSizePolicy::Fixed, TQSizePolicy::Fixed));
topicLine->setSizePolicy(TQSizePolicy(TQSizePolicy::Preferred, TQSizePolicy::Minimum));
commandLineBox->setSizePolicy(vfixed);
limit->setMaximumSize(40,100);
limit->setSizePolicy(hfixed);
modeT->setMaximumSize(20,100);
modeT->setSizePolicy(hfixed);
modeN->setMaximumSize(20,100);
modeN->setSizePolicy(hfixed);
modeS->setMaximumSize(20,100);
modeS->setSizePolicy(hfixed);
modeI->setMaximumSize(20,100);
modeI->setSizePolicy(hfixed);
modeP->setMaximumSize(20,100);
modeP->setSizePolicy(hfixed);
modeM->setMaximumSize(20,100);
modeM->setSizePolicy(hfixed);
modeK->setMaximumSize(20,100);
modeK->setSizePolicy(hfixed);
modeL->setMaximumSize(20,100);
modeL->setSizePolicy(hfixed);
getTextView()->setSizePolicy(greedy);
nicknameListView->setSizePolicy(hmodest);
connect(channelInput,TQT_SIGNAL (submit()),this,TQT_SLOT (channelTextEntered()) );
connect(channelInput,TQT_SIGNAL (envelopeCommand()),this,TQT_SLOT (channelPassthroughCommand()) );
connect(channelInput,TQT_SIGNAL (nickCompletion()),this,TQT_SLOT (completeNick()) );
connect(channelInput,TQT_SIGNAL (endCompletion()),this,TQT_SLOT (endCompleteNick()) );
connect(channelInput,TQT_SIGNAL (textPasted(const TQString&)),this,TQT_SLOT (textPasted(const TQString&)) );
connect(getTextView(), TQT_SIGNAL(textPasted(bool)), channelInput, TQT_SLOT(paste(bool)));
connect(getTextView(),TQT_SIGNAL (gotFocus()),channelInput,TQT_SLOT (setFocus()) );
connect(getTextView(),TQT_SIGNAL (sendFile()),this,TQT_SLOT (sendFileMenu()) );
connect(getTextView(),TQT_SIGNAL (autoText(const TQString&)),this,TQT_SLOT (sendChannelText(const TQString&)) );
connect(nicknameListView,TQT_SIGNAL (popupCommand(int)),this,TQT_SLOT (popupCommand(int)) );
connect(nicknameListView,TQT_SIGNAL (doubleClicked(TQListViewItem*)),this,TQT_SLOT (doubleClickCommand(TQListViewItem*)) );
connect(nicknameListView,TQT_SIGNAL (dropped(TQDropEvent*,TQListViewItem*)),this,TQT_SLOT (filesDropped(TQDropEvent*)) );
connect(nicknameCombobox,TQT_SIGNAL (activated(int)),this,TQT_SLOT(nicknameComboboxChanged()));
if(nicknameCombobox->lineEdit())
connect(nicknameCombobox->lineEdit(), TQT_SIGNAL (lostFocus()),this,TQT_SLOT(nicknameComboboxChanged()));
nicknameList.setAutoDelete(true);
setLog(Preferences::log());
connect(&userhostTimer,TQT_SIGNAL (timeout()),this,TQT_SLOT (autoUserhost()));
// every few seconds try to get more userhosts
userhostTimer.start(10000);
connect(&m_whoTimer,TQT_SIGNAL (timeout()),this,TQT_SLOT (autoWho()));
// every 5 minutes decrease everyone's activity by 1 unit
m_fadeActivityTimer.start(5*60*1000);
connect(&m_fadeActivityTimer, TQT_SIGNAL(timeout()), this, TQT_SLOT(fadeActivity()));
// re-schedule when the settings were changed
connect(Preferences::self(), TQT_SIGNAL (autoContinuousWhoChanged()),this,TQT_SLOT (scheduleAutoWho()));
updateAppearance();
//FIXME JOHNFLUX
// connect( Konversation::Addressbook::self()->getAddressBook(), TQT_SIGNAL( addressBookChanged( AddressBook * ) ), this, TQT_SLOT( slotLoadAddressees() ) );
// connect( Konversation::Addressbook::self(), TQT_SIGNAL(addresseesChanged()), this, TQT_SLOT(slotLoadAddressees()));
}
//FIXME there is some logic in setLogfileName that needs to be split out and called here if the server display name gets changed
void Channel::setServer(Server* server)
{
if (m_server != server)
connect(server, TQT_SIGNAL(connectionStateChanged(Server*, Konversation::ConnectionState)),
TQT_SLOT(connectionStateChanged(Server*, Konversation::ConnectionState)));
ChatWindow::setServer(server);
if (!server->getKeyForRecipient(getName()).isNull())
blowfishLabel->show();
topicLine->setServer(server);
refreshModeButtons();
setIdentity(server->getIdentity());
}
void Channel::connectionStateChanged(Server* server, Konversation::ConnectionState state)
{
if (server == m_server)
{
if (state != Konversation::SSConnected)
{
m_joined = false;
//HACK the way the notification priorities work sucks, this forces the tab text color to gray right now.
if (m_currentTabNotify == Konversation::tnfNone || (!Preferences::tabNotificationsEvents() && m_currentTabNotify == Konversation::tnfControl))
KonversationApplication::instance()->getMainWindow()->getViewContainer()->unsetViewNotification(this);
}
}
}
void Channel::setEncryptedOutput(bool e)
{
if (e) {
blowfishLabel->show();
//scan the channel topic and decrypt it if necessary
TQString topic(m_topicHistory[0].section(' ',2));
//prepend two colons to make it appear to be an irc message for decryption,
// \r because it won't decrypt without it, even though the message did not have a \r
// when encrypted. Bring on the TQCA!
TQCString cipher="::"+topic.utf8()+'\x0d';
Konversation::decryptTopic(getName(), cipher, m_server);
topic=TQString::fromUtf8(cipher.data()+2, cipher.length()-2);
m_topicHistory[0] = m_topicHistory[0].section(' ', 0, 1) + ' ' + topic;
topicLine->setText(topic);
emit topicHistoryChanged();
}
else
blowfishLabel->hide();
}
Channel::~Channel()
{
kdDebug() << "Channel::~Channel(" << getName() << ")" << endl;
// Purge nickname list
purgeNicks();
kdDebug() << "nicks purged" << endl;
// Unlink this channel from channel list
m_server->removeChannel(this);
kdDebug() << "Channel removed." << endl;
}
bool Channel::rejoinable()
{
if (getServer() && getServer()->isConnected())
return !m_joined;
return false;
}
void Channel::rejoin()
{
if (rejoinable())
m_server->sendJoinCommand(getName(), getPassword());
}
ChannelNickPtr Channel::getOwnChannelNick()
{
return m_ownChannelNick;
}
ChannelNickPtr Channel::getChannelNick(const TQString &ircnick)
{
return m_server->getChannelNick(getName(), ircnick);
}
void Channel::purgeNicks()
{
// Purge nickname list
nicknameList.clear();
// clear stats counter
nicks=0;
ops=0;
}
void Channel::showOptionsDialog()
{
if(!m_optionsDialog)
m_optionsDialog = new Konversation::ChannelOptionsDialog(this);
m_optionsDialog->refreshModes();
m_optionsDialog->refreshTopicHistory();
m_optionsDialog->show();
}
void Channel::filesDropped(TQDropEvent* e)
{
TQPoint p(nicknameListView->contentsToViewport(e->pos()));
Nick* it = dynamic_cast<Nick*>(nicknameListView->itemAt(p));
if (!it) return;
TQStrList uris;
if (TQUriDrag::decode(e,uris))
m_server->sendURIs(uris, it->getChannelNick()->getNickname());
}
void Channel::textPasted(const TQString& text)
{
if(m_server)
{
TQStringList multiline=TQStringList::split('\n',text);
for(unsigned int index=0;index<multiline.count();index++)
{
TQString line=multiline[index];
TQString cChar(Preferences::commandChar());
// make sure that lines starting with command char get escaped
if(line.startsWith(cChar)) line=cChar+line;
sendChannelText(line);
}
}
}
// Will be connected to IRCView::popupCommand(int)
void Channel::popupChannelCommand(int id)
{
channelCommand = true; // Context menu executed from ircview
popupCommand(id);
textView->clearContextNick();
channelCommand = false;
}
// Will be connected to NickListView::popupCommand(int)
void Channel::popupCommand(int id)
{
TQString pattern;
TQString cc = Preferences::commandChar();
TQString args;
TQString question;
bool raw=false;
TQStringList nickList = getSelectedNickList();
switch(id)
{
case Konversation::AddressbookEdit:
{
ChannelNickList nickList=getSelectedChannelNicks();
for(ChannelNickList::ConstIterator it=nickList.begin();it!=nickList.end();++it)
{
if(!(*it)->getNickInfo()->editAddressee()) break;;
}
break;
}
case Konversation::AddressbookNew:
case Konversation::AddressbookDelete:
{
Konversation::Addressbook *addressbook = Konversation::Addressbook::self();
ChannelNickList nickList=getSelectedChannelNicks();
//Handle all the selected nicks in one go. Either they all save, or none do.
if(addressbook->getAndCheckTicket())
{
for(ChannelNickList::ConstIterator it=nickList.begin();it!=nickList.end();++it)
{
if(id == Konversation::AddressbookDelete)
{
KABC::Addressee addr = (*it)->getNickInfo()->getAddressee();
addressbook->unassociateNick(addr, (*it)->getNickname(), m_server->getServerName(), m_server->getDisplayName());
}
else
{
//make new addressbook contact
KABC::Addressee addr;
NickInfoPtr nickInfo = (*it)->getNickInfo();
if(nickInfo->getRealName().isEmpty())
addr.setGivenName(nickInfo->getNickname());
else
addr.setGivenName(nickInfo->getRealName());
addr.setNickName(nickInfo->getNickname());
addressbook->associateNickAndUnassociateFromEveryoneElse(addr, (*it)->getNickname(), m_server->getServerName(), m_server->getDisplayName());
}
}
addressbook->saveTicket(); // This will refresh the nicks automatically for us. At least, if it doesn't, it's a bug :)
}
break;
}
case Konversation::AddressbookChange:
{
ChannelNickList nickList=getSelectedChannelNicks();
for(ChannelNickList::ConstIterator it=nickList.begin();it!=nickList.end();++it)
{
(*it)->getNickInfo()->showLinkAddressbookUI();
}
break;
}
case Konversation::SendEmail:
{
Konversation::Addressbook::self()->sendEmail(getSelectedChannelNicks());
break;
}
case Konversation::AddressbookSub:
kdDebug() << "sub called" << endl;
break;
case Konversation::GiveOp:
pattern="MODE %c +o %u";
raw=true;
break;
case Konversation::TakeOp:
pattern="MODE %c -o %u";
raw=true;
break;
case Konversation::GiveHalfOp:
pattern="MODE %c +h %u";
raw=true;
break;
case Konversation::TakeHalfOp:
pattern="MODE %c -h %u";
raw=true;
break;
case Konversation::GiveVoice:
pattern="MODE %c +v %u";
raw=true;
break;
case Konversation::TakeVoice:
pattern="MODE %c -v %u";
raw=true;
break;
case Konversation::Version:
pattern=cc+"CTCP %u VERSION";
break;
case Konversation::Whois:
pattern="WHOIS %u %u";
raw=true;
break;
case Konversation::Topic:
m_server->requestTopic(getTextView()->currentChannel());
break;
case Konversation::Names:
m_server->queue("NAMES " + getTextView()->currentChannel(), Server::LowPriority);
break;
case Konversation::Join:
m_server->queue("JOIN " + getTextView()->currentChannel());
break;
case Konversation::Ping:
{
unsigned int time_t = TQDateTime::currentDateTime().toTime_t();
pattern=TQString(cc+"CTCP %u PING %1").arg(time_t);
}
break;
case Konversation::Kick:
pattern=cc+"KICK %u";
break;
case Konversation::KickBan:
pattern=cc+"BAN %u\n"+
cc+"KICK %u";
break;
case Konversation::BanNick:
pattern=cc+"BAN %u";
break;
case Konversation::BanHost:
pattern=cc+"BAN -HOST %u";
break;
case Konversation::BanDomain:
pattern=cc+"BAN -DOMAIN %u";
break;
case Konversation::BanUserHost:
pattern=cc+"BAN -USERHOST %u";
break;
case Konversation::BanUserDomain:
pattern=cc+"BAN -USERDOMAIN %u";
break;
case Konversation::KickBanHost:
pattern=cc+"KICKBAN -HOST %u";
break;
case Konversation::KickBanDomain:
pattern=cc+"KICKBAN -DOMAIN %u";
break;
case Konversation::KickBanUserHost:
pattern=cc+"KICKBAN -USERHOST %u";
break;
case Konversation::KickBanUserDomain:
pattern=cc+"KICKBAN -USERDOMAIN %u";
break;
case Konversation::OpenQuery:
pattern=cc+"QUERY %u";
break;
case Konversation::StartDccChat:
pattern=cc+"DCC CHAT %u";
break;
case Konversation::DccSend:
pattern=cc+"DCC SEND %u";
break;
case Konversation::IgnoreNick:
if (nickList.size() == 1)
question=i18n("Do you want to ignore %1?").arg(nickList.first());
else
question = i18n("Do you want to ignore the selected users?");
if (KMessageBox::warningContinueCancel(this, question, i18n("Ignore"), i18n("Ignore"), "IgnoreNick") ==
KMessageBox::Continue)
pattern = cc+"IGNORE -ALL %l";
break;
case Konversation::UnignoreNick:
{
TQStringList selectedIgnoredNicks;
for (TQStringList::Iterator it=nickList.begin(); it!=nickList.end(); ++it)
{
if (Preferences::isIgnored((*it)))
selectedIgnoredNicks.append((*it));
}
if (selectedIgnoredNicks.count() == 1)
question=i18n("Do you want to stop ignoring %1?").arg(selectedIgnoredNicks.first());
else
question = i18n("Do you want to stop ignoring the selected users?");
if (KMessageBox::warningContinueCancel(this, question, i18n("Unignore"), i18n("Unignore"), "UnignoreNick") ==
KMessageBox::Continue)
{
sendChannelText(cc+"UNIGNORE "+selectedIgnoredNicks.join(" "));
}
break;
}
case Konversation::AddNotify:
{
if (m_server->getServerGroup())
{
for (TQStringList::Iterator it=nickList.begin(); it!=nickList.end(); ++it)
{
if (!Preferences::isNotify(m_server->getServerGroup()->id(), (*it)))
Preferences::addNotify(m_server->getServerGroup()->id(), (*it));
}
}
break;
}
} // switch
if (!pattern.isEmpty())
{
pattern.replace("%c",getName());
TQString command;
if (pattern.contains("%l"))
{
TQStringList list;
for (TQStringList::Iterator it=nickList.begin(); it!=nickList.end(); ++it)
list.append((*it));
command = pattern.replace("%l", list.join(" "));
if (raw)
m_server->queue(command);
else
sendChannelText(command);
}
else
{
TQStringList patternList = TQStringList::split('\n',pattern);
for (TQStringList::Iterator it=nickList.begin(); it!=nickList.end(); ++it)
{
for (unsigned int index = 0; index<patternList.count(); index++)
{
command = patternList[index];
command.replace("%u", (*it));
if (raw)
m_server->queue(command);
else
sendChannelText(command);
}
}
}
}
}
// Will be connected to NickListView::doubleClicked()
void Channel::doubleClickCommand(TQListViewItem* item)
{
if(item)
{
nicknameListView->clearSelection();
nicknameListView->setSelected(item, true);
// TODO: put the quick button code in another function to make reusal more legitimate
quickButtonClicked(Preferences::channelDoubleClickAction());
}
}
void Channel::completeNick()
{
int pos, oldPos;
channelInput->getCursorPosition(&oldPos,&pos);// oldPos is a dummy here, taking the paragraph parameter
oldPos = channelInput->getOldCursorPosition();
TQString line=channelInput->text();
TQString newLine;
// Check if completion position is out of range
if(completionPosition >= nicknameList.count()) completionPosition = 0;
// Check, which completion mode is active
char mode = channelInput->getCompletionMode();
if(mode == 'c')
{
line.remove(oldPos, pos - oldPos);
pos = oldPos;
}
// If the cursor is at beginning of line, insert last completion
if(pos == 0 && !channelInput->lastCompletion().isEmpty())
{
TQString addStart(Preferences::nickCompleteSuffixStart());
newLine = channelInput->lastCompletion() + addStart;
// New cursor position is behind nickname
pos = newLine.length();
// Add rest of the line
newLine += line;
}
else
{
// remember old cursor position in input field
channelInput->setOldCursorPosition(pos);
// remember old cursor position locally
oldPos = pos;
// step back to last space or start of line
while(pos && line[pos-1] != ' ') pos--;
// copy search pattern (lowercase)
TQString pattern = line.mid(pos, oldPos - pos);
// copy line to newLine-buffer
newLine = line;
// did we find any pattern?
if(!pattern.isEmpty())
{
bool complete = false;
TQString foundNick;
// try to find matching nickname in list of names
if(Preferences::nickCompletionMode() == 1 ||
Preferences::nickCompletionMode() == 2)
{ // Shell like completion
TQStringList found;
foundNick = nicknameList.completeNick(pattern, complete, found,
(Preferences::nickCompletionMode() == 2),
Preferences::nickCompletionCaseSensitive());
if(!complete && !found.isEmpty())
{
if(Preferences::nickCompletionMode() == 1)
{
TQString nicksFound = found.join(" ");
appendServerMessage(i18n("Completion"), i18n("Possible completions: %1.").arg(nicksFound));
}
else
{
channelInput->showCompletionList(found);
}
}
} // Cycle completion
else if(Preferences::nickCompletionMode() == 0)
{
if(mode == '\0') {
TQPtrListIterator<Nick> it(nicknameList);
uint timeStamp = 0;
int listPosition = 0;
Nick* nick = 0;
while(it.current() != 0)
{
nick = it.current();
if(nick->getChannelNick()->getNickname().startsWith(pattern, Preferences::nickCompletionCaseSensitive()) &&
(nick->getChannelNick()->timeStamp() > timeStamp))
{
timeStamp = nick->getChannelNick()->timeStamp();
completionPosition = listPosition;
}
++listPosition;
++it;
}
}
// remember old nick completion position
unsigned int oldCompletionPosition = completionPosition;
complete = true;
TQString prefixCharacter = Preferences::prefixCharacter();
do
{
TQString lookNick = nicknameList.at(completionPosition)->getChannelNick()->getNickname();
if(!prefixCharacter.isEmpty() && lookNick.contains(prefixCharacter))
{
lookNick = lookNick.section( prefixCharacter,1 );
}
if(lookNick.startsWith(pattern, Preferences::nickCompletionCaseSensitive()))
{
foundNick = lookNick;
}
// increment search position
completionPosition++;
// wrap around
if(completionPosition == nicknameList.count())
{
completionPosition = 0;
}
// the search ends when we either find a suitable nick or we end up at the
// first search position
} while((completionPosition != oldCompletionPosition) && foundNick.isEmpty());
}
// did we find a suitable nick?
if(!foundNick.isEmpty())
{
// set channel nicks completion mode
channelInput->setCompletionMode('c');
// remove pattern from line
newLine.remove(pos, pattern.length());
// did we find the nick in the middle of the line?
if(pos && complete)
{
channelInput->setLastCompletion(foundNick);
TQString addMiddle = Preferences::nickCompleteSuffixMiddle();
newLine.insert(pos, foundNick + addMiddle);
pos = pos + foundNick.length() + addMiddle.length();
}
// no, it was at the beginning
else if(complete)
{
channelInput->setLastCompletion(foundNick);
TQString addStart = Preferences::nickCompleteSuffixStart();
newLine.insert(pos, foundNick + addStart);
pos = pos + foundNick.length() + addStart.length();
}
// the nick wasn't complete
else
{
newLine.insert(pos, foundNick);
pos = pos + foundNick.length();
}
}
// no pattern found, so restore old cursor position
else pos = oldPos;
}
}
// Set new text and cursor position
channelInput->setText(newLine);
channelInput->setCursorPosition(0, pos);
}
// make sure to step back one position when completion ends so the user starts
// with the last complete they made
void Channel::endCompleteNick()
{
if(completionPosition) completionPosition--;
else completionPosition=nicknameList.count()-1;
}
void Channel::setName(const TQString& newName)
{
ChatWindow::setName(newName);
setLogfileName(newName.lower());
}
bool Channel::autoJoin()
{
if (!m_server->getServerGroup()) return false;
Konversation::ChannelList channelList = m_server->getServerGroup()->channelList();
if (!channelList.empty())
return channelList.find(channelSettings()) != channelList.end();
else
return false;
}
void Channel::setAutoJoin(bool autojoin)
{
if (autojoin && !(autoJoin()))
{
Konversation::ChannelSettings before;
TQPtrList<Channel> channelList = m_server->getChannelList();
if (channelList.count() > 1)
{
TQPtrListIterator<Channel> it(channelList);
Channel* channel;
TQMap<int, Channel*> channelMap;
int index = -1;
int ownIndex = m_server->getViewContainer()->getViewIndex(this);
while ((channel = it.current()) != 0)
{
++it;
index = m_server->getViewContainer()->getViewIndex(channel);
if (index && index > ownIndex) channelMap.insert(index, channel);
}
if (channelMap.count())
{
TQMap<int, Channel*>::Iterator it2;
for (it2 = channelMap.begin(); it2 != channelMap.end(); ++it2)
{
channel = it2.data();
if (channel->autoJoin())
{
before = channel->channelSettings();
break;
}
}
}
}
if (m_server->getServerGroup())
m_server->getServerGroup()->addChannel(channelSettings(), before);
}
else
{
if (m_server->getServerGroup())
m_server->getServerGroup()->removeChannel(channelSettings());
}
}
TQString Channel::getPassword()
{
TQString password;
for (TQStringList::const_iterator it = m_modeList.begin(); it != m_modeList.end(); ++it)
{
if ((*it)[0] == 'k') password = (*it).mid(1);
}
return password;
}
const Konversation::ChannelSettings Channel::channelSettings()
{
Konversation::ChannelSettings channel;
channel.setName(getName());
channel.setPassword(getPassword());
channel.setNotificationsEnabled(notificationsEnabled());
return channel;
}
void Channel::sendFileMenu()
{
emit sendFile();
}
void Channel::channelTextEntered()
{
TQString line = channelInput->text();
channelInput->setText ("");
if(line.lower().stripWhiteSpace() == Preferences::commandChar()+"clear")
{
textView->clear();
}
else if(line.lower().stripWhiteSpace() == Preferences::commandChar()+"cycle")
{
cycleChannel();
}
else
{
if(!line.isEmpty())
sendChannelText(line);
}
}
void Channel::channelPassthroughCommand()
{
TQString commandChar = Preferences::commandChar();
TQString line = channelInput->text();
channelInput->setText("");
if(!line.isEmpty())
{
// Prepend commandChar on Ctrl+Enter to bypass outputfilter command recognition
if (line.startsWith(commandChar))
{
line = commandChar + line;
}
sendChannelText(line);
}
}
void Channel::sendChannelText(const TQString& sendLine)
{
// create a work copy
TQString outputAll(sendLine);
// replace aliases and wildcards
if(m_server->getOutputFilter()->replaceAliases(outputAll))
{
outputAll = m_server->parseWildcards(outputAll,m_server->getNickname(),getName(),getPassword(),
getSelectedNickList(),TQString());
}
// Send all strings, one after another
TQStringList outList=TQStringList::split(TQRegExp("[\r\n]+"),outputAll);
for(unsigned int index=0;index<outList.count();index++)
{
TQString output(outList[index]);
// encoding stuff is done in Server()
Konversation::OutputFilterResult result = m_server->getOutputFilter()->parse(m_server->getNickname(),output,getName());
// Is there something we need to display for ourselves?
if(!result.output.isEmpty())
{
if(result.type == Konversation::Action) appendAction(m_server->getNickname(), result.output);
else if(result.type == Konversation::Command) appendCommandMessage(result.typeString, result.output);
else if(result.type == Konversation::Program) appendServerMessage(result.typeString, result.output);
else if(result.type == Konversation::PrivateMessage) appendQuery(result.typeString, result.output, true);
else append(m_server->getNickname(), result.output);
}
else if (result.outputList.count())
{
Q_ASSERT(result.type==Konversation::Message);
for ( TQStringList::Iterator it = result.outputList.begin(); it != result.outputList.end(); ++it )
{
append(m_server->getNickname(), *it);
}
}
// Send anything else to the server
if(!result.toServerList.empty())
{
m_server->queueList(result.toServerList);
}
else
{
m_server->queue(result.toServer);
}
} // for
}
void Channel::setNickname(const TQString& newNickname)
{
nicknameCombobox->setCurrentText(newNickname);
}
TQStringList Channel::getSelectedNickList()
{
TQStringList result;
if (channelCommand)
result.append(textView->getContextNick());
else
{
Nick* nick=nicknameList.first();
while(nick)
{
if(nick->isSelected()) result.append(nick->getChannelNick()->getNickname());
nick=nicknameList.next();
}
}
return result;
}
ChannelNickList Channel::getSelectedChannelNicks()
{
ChannelNickList result;
Nick* nick=nicknameList.first();
while(nick)
{
if(channelCommand)
{
if(nick->getChannelNick()->getNickname() == textView->getContextNick())
{
result.append(nick->getChannelNick());
return result;
}
}
else if(nick->isSelected())
result.append(nick->getChannelNick());
nick=nicknameList.next();
}
return result;
}
void Channel::channelLimitChanged()
{
unsigned int lim=limit->text().toUInt();
modeButtonClicked(7,lim>0);
}
void Channel::modeButtonClicked(int id, bool on)
{
char mode[]={'t','n','s','i','p','m','k','l'};
TQString command("MODE %1 %2%3 %4");
TQString args = getPassword();
if (mode[id] == 'k')
{
if (args.isEmpty())
{
TQCString newPassword;
int result = KPasswordDialog::getPassword(newPassword, i18n("Channel Password"));
if (result == KPasswordDialog::Accepted && !newPassword.isEmpty())
args = newPassword;
}
}
else if(mode[id]=='l')
{
if(limit->text().isEmpty() && on)
{
bool ok=false;
// ask user how many nicks should be the limit
args=KInputDialog::getText(i18n("Nick Limit"),
i18n("Enter the new nick limit:"),
limit->text(), // will be always "" but what the hell ;)
&ok,
this);
// leave this function if user cancels
if(!ok) return;
}
else if(on)
args=limit->text();
}
// put together the mode command and send it to the server queue
m_server->queue(command.arg(getName()).arg((on) ? "+" : "-").arg(mode[id]).arg(args));
}
void Channel::quickButtonClicked(const TQString &buttonText)
{
// parse wildcards (toParse,nickname,channelName,nickList,queryName,parameter)
TQString out=m_server->parseWildcards(buttonText,m_server->getNickname(),getName(),getPassword(),getSelectedNickList(),TQString());
// are there any newlines in the definition?
if(out.find('\n')!=-1)
sendChannelText(out);
// single line without newline needs to be copied into input line
else
channelInput->setText(out);
}
void Channel::addNickname(ChannelNickPtr channelnick)
{
TQString nickname = channelnick->loweredNickname();
Nick* nick=0;
Nick* lookNick;
TQPtrListIterator<Nick> it(nicknameList);
while((lookNick = it.current()) != 0)
{
if(lookNick->getChannelNick()->loweredNickname() == nickname)
{
nick = lookNick;
break;
}
++it;
}
if(nick == 0)
{
fastAddNickname(channelnick);
if(channelnick->isAnyTypeOfOp())
{
adjustOps(1);
}
adjustNicks(1);
requestNickListSort();
}
else
{
Q_ASSERT(false); // We shouldn't be adding someone that is already in the channel.
}
}
// Use with caution! Does not sort or check for duplicates!
void Channel::fastAddNickname(ChannelNickPtr channelnick)
{
Q_ASSERT(channelnick);
if(!channelnick) return;
Nick* nick = new Nick(nicknameListView, channelnick);
// nicks get sorted later
nicknameList.append(nick);
}
void Channel::nickRenamed(const TQString &oldNick, const NickInfo& nickInfo)
{
/* Did we change our nick name? */
TQString newNick = nickInfo.getNickname();
if(newNick == m_server->getNickname()) /* Check newNick because m_server->getNickname() is already updated to new nick */
{
setNickname(newNick);
appendCommandMessage(i18n("Nick"),i18n("You are now known as %1.").arg(newNick), false, true, true);
}
else
{
/* No, must've been someone else */
appendCommandMessage(i18n("Nick"),i18n("%1 is now known as %2.").arg(oldNick).arg(newNick),false);
}
nicknameListView->sort();
}
void Channel::joinNickname(ChannelNickPtr channelNick)
{
if(channelNick->getNickname() == m_server->getNickname())
{
m_joined = true;
emit joined(this);
appendCommandMessage(i18n("Join"),i18n("%1 is the channel and %2 is our hostmask",
"You have joined the channel %1 (%2).").arg(getName()).arg(channelNick->getHostmask()),false, false, true);
m_ownChannelNick = channelNick;
connect(m_ownChannelNick, TQT_SIGNAL(channelNickChanged()), TQT_SLOT(refreshModeButtons()));
refreshModeButtons();
setActive(true);
//HACK the way the notification priorities work sucks, this forces the tab text color to ungray right now.
if (m_currentTabNotify == Konversation::tnfNone || !Preferences::tabNotificationsEvents())
KonversationApplication::instance()->getMainWindow()->getViewContainer()->unsetViewNotification(this);
KonversationApplication::instance()->notificationHandler()->channelJoin(this,getName());
}
else
{
TQString nick = channelNick->getNickname();
TQString hostname = channelNick->getHostmask();
appendCommandMessage(i18n("Join"), i18n("%1 is the nick joining and %2 the hostmask of that nick",
"%1 has joined this channel (%2).").arg(nick).arg(hostname),false, false);
addNickname(channelNick);
}
}
void Channel::removeNick(ChannelNickPtr channelNick, const TQString &reason, bool quit)
{
TQString displayReason = reason;
if(!displayReason.isEmpty())
{
// if the reason contains text markup characters, play it safe and reset all
if(displayReason.find(TQRegExp("[\\0000-\\0037]")) != -1)
displayReason += "\017";
}
if(channelNick->getNickname() == m_server->getNickname())
{
//If in the future we can leave a channel, but not close the window, refreshModeButtons() has to be called.
if (quit)
{
if (displayReason.isEmpty())
appendCommandMessage(i18n("Quit"), i18n("You have left this server."), false);
else
appendCommandMessage(i18n("Quit"), i18n("%1 adds the reason", "You have left this server (%1).").arg(displayReason), false);
}
else
{
if (displayReason.isEmpty())
appendCommandMessage(i18n("Part"), i18n("You have left channel %1.").arg(getName()), false);
else
appendCommandMessage(i18n("Part"), i18n("%1 adds the channel and %2 the reason",
"You have left channel %1 (%2).").arg(getName()).arg(displayReason), false);
}
delete this;
}
else
{
if (quit)
{
if (displayReason.isEmpty())
appendCommandMessage(i18n("Quit"), i18n("%1 has left this server.").arg(channelNick->getNickname()), false);
else
appendCommandMessage(i18n("Quit"), i18n("%1 adds the nick and %2 the reason",
"%1 has left this server (%2).").arg(channelNick->getNickname()).arg(displayReason), false);
}
else
{
if (displayReason.isEmpty())
appendCommandMessage(i18n("Part"), i18n("%1 has left this channel.").arg(channelNick->getNickname()), false);
else
appendCommandMessage(i18n("Part"), i18n("%1 adds the nick and %2 the reason",
"%1 has left this channel (%2).").arg(channelNick->getNickname()).arg(displayReason), false);
}
if(channelNick->isAnyTypeOfOp())
{
adjustOps(-1);
}
adjustNicks(-1);
Nick* nick = getNickByName(channelNick->getNickname());
if(nick)
{
nicknameList.removeRef(nick);
}
else
{
kdWarning() << "Channel::removeNick(): Nickname " << channelNick->getNickname() << " not found!"<< endl;
}
}
}
void Channel::kickNick(ChannelNickPtr channelNick, const TQString &kicker, const TQString &reason)
{
TQString displayReason = reason;
if(!displayReason.isEmpty())
{
// if the reason contains text markup characters, play it safe and reset all
if(displayReason.find(TQRegExp("[\\0000-\\0037]")) != -1)
displayReason += "\017";
}
if(channelNick->getNickname() == m_server->getNickname())
{
if(kicker == m_server->getNickname())
{
if (displayReason.isEmpty())
appendCommandMessage(i18n("Kick"), i18n("You have kicked yourself from channel %1.").arg(getName()));
else
appendCommandMessage(i18n("Kick"), i18n("%1 adds the channel and %2 the reason",
"You have kicked yourself from channel %1 (%2).").arg(getName()).arg(displayReason));
}
else
{
if (displayReason.isEmpty())
{
appendCommandMessage(i18n("Kick"), i18n("%1 adds the channel, %2 adds the kicker",
"You have been kicked from channel %1 by %2.")
.arg(getName()).arg(kicker), true);
}
else
{
appendCommandMessage(i18n("Kick"), i18n("%1 adds the channel, %2 the kicker and %3 the reason",
"You have been kicked from channel %1 by %2 (%3).")
.arg(getName()).arg(kicker).arg(displayReason), true);
}
KonversationApplication::instance()->notificationHandler()->kick(this,getName(), kicker);
}
m_joined=false;
setActive(false);
//HACK the way the notification priorities work sucks, this forces the tab text color to gray right now.
if (m_currentTabNotify == Konversation::tnfNone || (!Preferences::tabNotificationsEvents() && m_currentTabNotify == Konversation::tnfControl))
KonversationApplication::instance()->getMainWindow()->getViewContainer()->unsetViewNotification(this);
return;
}
else
{
if(kicker == m_server->getNickname())
{
if (displayReason.isEmpty())
appendCommandMessage(i18n("Kick"), i18n("You have kicked %1 from the channel.").arg(channelNick->getNickname()));
else
appendCommandMessage(i18n("Kick"), i18n("%1 adds the kicked nick and %2 the reason",
"You have kicked %1 from the channel (%2).").arg(channelNick->getNickname()).arg(displayReason), true);
}
else
{
if (displayReason.isEmpty())
{
appendCommandMessage(i18n("Kick"), i18n("%1 adds the kicked nick, %2 adds the kicker",
"%1 has been kicked from the channel by %2.")
.arg(channelNick->getNickname()).arg(kicker), true);
}
else
{
appendCommandMessage(i18n("Kick"), i18n("%1 adds the kicked nick, %2 the kicker and %3 the reason",
"%1 has been kicked from the channel by %2 (%3).")
.arg(channelNick->getNickname()).arg(kicker).arg(displayReason), true);
}
}
if(channelNick->isAnyTypeOfOp())
adjustOps(-1);
adjustNicks(-1);
Nick* nick = getNickByName(channelNick->getNickname());
if(nick == 0)
{
kdWarning() << "Channel::kickNick(): Nickname " << channelNick->getNickname() << " not found!"<< endl;
}
else
{
nicknameList.removeRef(nick);
}
}
}
Nick* Channel::getNickByName(const TQString &lookname)
{
TQString lcLookname = lookname.lower();
TQPtrListIterator<Nick> it(nicknameList);
while(it.current() != 0)
{
if(it.current()->getChannelNick()->loweredNickname() == lcLookname)
return it.current();
++it;
}
return 0;
}
void Channel::adjustNicks(int value)
{
if((nicks == 0) && (value <= 0))
{
return;
}
nicks += value;
if(nicks < 0)
{
nicks = 0;
}
emitUpdateInfo();
}
void Channel::adjustOps(int value)
{
if((ops == 0) && (value <= 0))
{
return;
}
ops += value;
if(ops < 0)
{
ops = 0;
}
emitUpdateInfo();
}
void Channel::emitUpdateInfo()
{
TQString info = getName() + " - ";
info += i18n("%n nick", "%n nicks", numberOfNicks());
info += i18n(" (%n op)", " (%n ops)", numberOfOps());
emit updateInfo(info);
}
void Channel::setTopic(const TQString &newTopic)
{
appendCommandMessage(i18n("Topic"), i18n("The channel topic is \"%1\".").arg(newTopic));
TQString topic = Konversation::removeIrcMarkup(newTopic);
topicLine->setText(topic);
topicAuthorUnknown=true; // if we only get called with a topic, it was a 332, which usually has a 333 next
// cut off "nickname" and "time_t" portion of the topic before comparing, otherwise the history
// list will fill up with the same entries while the user only requests the topic to be seen.
if(m_topicHistory.first().section(' ', 2) != newTopic)
{
m_topicHistory.prepend(TQString("%1 "+i18n("unknown")+" %2").arg(TQDateTime::currentDateTime().toTime_t()).arg(newTopic));
emit topicHistoryChanged();
}
}
void Channel::setTopic(const TQString &nickname, const TQString &newTopic) // Overloaded
{
if(nickname == m_server->getNickname())
{
appendCommandMessage(i18n("Topic"), i18n("You set the channel topic to \"%1\".").arg(newTopic));
}
else
{
appendCommandMessage(i18n("Topic"), i18n("%1 sets the channel topic to \"%2\".").arg(nickname).arg(newTopic));
}
m_topicHistory.prepend(TQString("%1 %2 %3").arg(TQDateTime::currentDateTime().toTime_t()).arg(nickname).arg(newTopic));
TQString topic = Konversation::removeIrcMarkup(newTopic);
topicLine->setText(topic);
emit topicHistoryChanged();
}
TQStringList Channel::getTopicHistory()
{
return m_topicHistory;
}
TQString Channel::getTopic()
{
return m_topicHistory[0];
}
void Channel::setTopicAuthor(const TQString& newAuthor, TQDateTime time)
{
if (time.isNull() || !time.isValid())
time=TQDateTime::currentDateTime();
if(topicAuthorUnknown)
{
m_topicHistory[0] = TQString("%1").arg(time.toTime_t()) + ' ' + newAuthor + ' ' + m_topicHistory[0].section(' ', 2);
topicAuthorUnknown = false;
emit topicHistoryChanged();
}
}
void Channel::updateMode(const TQString& sourceNick, char mode, bool plus, const TQString &parameter)
{
//Note for future expansion: doing m_server->getChannelNick(getName(), sourceNick); may not return a valid channelNickPtr if the
//mode is updated by the server.
TQString message;
ChannelNickPtr parameterChannelNick=m_server->getChannelNick(getName(), parameter);
bool fromMe=false;
bool toMe=false;
// remember if this nick had any type of op.
bool wasAnyOp=false;
if(parameterChannelNick)
wasAnyOp=parameterChannelNick->isAnyTypeOfOp();
if(sourceNick.lower()==m_server->loweredNickname())
fromMe=true;
if(parameter.lower()==m_server->loweredNickname())
toMe=true;
switch(mode)
{
case 'q':
if(plus)
{
if(fromMe)
{
if(toMe)
message=i18n("You give channel owner privileges to yourself.");
else
message=i18n("You give channel owner privileges to %1.").arg(parameter);
}
else
{
if(toMe)
message=i18n("%1 gives channel owner privileges to you.").arg(sourceNick);
else
message=i18n("%1 gives channel owner privileges to %2.").arg(sourceNick).arg(parameter);
}
}
else
{
if(fromMe)
{
if(toMe)
message=i18n("You take channel owner privileges from yourself.");
else
message=i18n("You take channel owner privileges from %1.").arg(parameter);
}
else
{
if(toMe)
message=i18n("%1 takes channel owner privileges from you.").arg(sourceNick);
else
message=i18n("%1 takes channel owner privileges from %2.").arg(sourceNick).arg(parameter);
}
}
if(parameterChannelNick)
{
parameterChannelNick->setOwner(plus);
emitUpdateInfo();
nicknameListView->sort();
}
break;
case 'a':
if(plus)
{
if(fromMe)
{
if(toMe)
message=i18n("You give channel admin privileges to yourself.");
else
message=i18n("You give channel admin privileges to %1.").arg(parameter);
}
else
{
if(toMe)
message=i18n("%1 gives channel admin privileges to you.").arg(sourceNick);
else
message=i18n("%1 gives channel admin privileges to %2.").arg(sourceNick).arg(parameter);
}
}
else
{
if(fromMe)
{
if(toMe)
message=i18n("You take channel admin privileges from yourself.");
else
message=i18n("You take channel admin privileges from %1.").arg(parameter);
}
else
{
if(toMe)
message=i18n("%1 takes channel admin privileges from you.").arg(sourceNick);
else
message=i18n("%1 takes channel admin privileges from %2.").arg(sourceNick).arg(parameter);
}
}
if(parameterChannelNick)
{
parameterChannelNick->setOwner(plus);
emitUpdateInfo();
nicknameListView->sort();
}
break;
case 'o':
if(plus)
{
if(fromMe)
{
if(toMe)
message=i18n("You give channel operator privileges to yourself.");
else
message=i18n("You give channel operator privileges to %1.").arg(parameter);
}
else
{
if(toMe)
message=i18n("%1 gives channel operator privileges to you.").arg(sourceNick);
else
message=i18n("%1 gives channel operator privileges to %2.").arg(sourceNick).arg(parameter);
}
}
else
{
if(fromMe)
{
if(toMe)
message=i18n("You take channel operator privileges from yourself.");
else
message=i18n("You take channel operator privileges from %1.").arg(parameter);
}
else
{
if(toMe)
message=i18n("%1 takes channel operator privileges from you.").arg(sourceNick);
else
message=i18n("%1 takes channel operator privileges from %2.").arg(sourceNick).arg(parameter);
}
}
if(parameterChannelNick)
{
parameterChannelNick->setOp(plus);
emitUpdateInfo();
nicknameListView->sort();
}
break;
case 'h':
if(plus)
{
if(fromMe)
{
if(toMe)
message=i18n("You give channel halfop privileges to yourself.");
else
message=i18n("You give channel halfop privileges to %1.").arg(parameter);
}
else
{
if(toMe)
message=i18n("%1 gives channel halfop privileges to you.").arg(sourceNick);
else
message=i18n("%1 gives channel halfop privileges to %2.").arg(sourceNick).arg(parameter);
}
}
else
{
if(fromMe)
{
if(toMe)
message=i18n("You take channel halfop privileges from yourself.");
else
message=i18n("You take channel halfop privileges from %1.").arg(parameter);
}
else
{
if(toMe)
message=i18n("%1 takes channel halfop privileges from you.").arg(sourceNick);
else
message=i18n("%1 takes channel halfop privileges from %2.").arg(sourceNick).arg(parameter);
}
}
if(parameterChannelNick)
{
parameterChannelNick->setHalfOp(plus);
emitUpdateInfo();
nicknameListView->sort();
}
break;
//case 'O': break;
case 'v':
if(plus)
{
if(fromMe)
{
if(toMe) message=i18n("You give yourself permission to talk.");
else message=i18n("You give %1 permission to talk.").arg(parameter);
}
else
{
if(toMe) message=i18n("%1 gives you permission to talk.").arg(sourceNick);
else message=i18n("%1 gives %2 permission to talk.").arg(sourceNick).arg(parameter);
}
}
else
{
if(fromMe)
{
if(toMe) message=i18n("You take the permission to talk from yourself.");
else message=i18n("You take the permission to talk from %1.").arg(parameter);
}
else
{
if(toMe) message=i18n("%1 takes the permission to talk from you.").arg(sourceNick);
else message=i18n("%1 takes the permission to talk from %2.").arg(sourceNick).arg(parameter);
}
}
if(parameterChannelNick)
{
parameterChannelNick->setVoice(plus);
nicknameListView->sort();
}
break;
case 'c':
if(plus)
{
if(fromMe) message=i18n("You set the channel mode to 'no colors allowed'.");
else message=i18n("%1 sets the channel mode to 'no colors allowed'.").arg(sourceNick);
}
else
{
if(fromMe) message=i18n("You set the channel mode to 'allow color codes'.");
else message=i18n("%1 sets the channel mode to 'allow color codes'.").arg(sourceNick);
}
break;
case 'i':
if(plus)
{
if(fromMe) message=i18n("You set the channel mode to 'invite only'.");
else message=i18n("%1 sets the channel mode to 'invite only'.").arg(sourceNick);
}
else
{
if(fromMe) message=i18n("You remove the 'invite only' mode from the channel.");
else message=i18n("%1 removes the 'invite only' mode from the channel.").arg(sourceNick);
}
modeI->setDown(plus);
break;
case 'm':
if(plus)
{
if(fromMe) message=i18n("You set the channel mode to 'moderated'.");
else message=i18n("%1 sets the channel mode to 'moderated'.").arg(sourceNick);
}
else
{
if(fromMe) message=i18n("You set the channel mode to 'unmoderated'.");
else message=i18n("%1 sets the channel mode to 'unmoderated'.").arg(sourceNick);
}
modeM->setDown(plus);
break;
case 'n':
if(plus)
{
if(fromMe) message=i18n("You set the channel mode to 'no messages from outside'.");
else message=i18n("%1 sets the channel mode to 'no messages from outside'.").arg(sourceNick);
}
else
{
if(fromMe) message=i18n("You set the channel mode to 'allow messages from outside'.");
else message=i18n("%1 sets the channel mode to 'allow messages from outside'.").arg(sourceNick);
}
modeN->setDown(plus);
break;
case 'p':
if(plus)
{
if(fromMe) message=i18n("You set the channel mode to 'private'.");
else message=i18n("%1 sets the channel mode to 'private'.").arg(sourceNick);
}
else
{
if(fromMe) message=i18n("You set the channel mode to 'public'.");
else message=i18n("%1 sets the channel mode to 'public'.").arg(sourceNick);
}
modeP->setDown(plus);
if(plus) modeS->setDown(false);
break;
case 's':
if(plus)
{
if(fromMe) message=i18n("You set the channel mode to 'secret'.");
else message=i18n("%1 sets the channel mode to 'secret'.").arg(sourceNick);
}
else
{
if(fromMe) message=i18n("You set the channel mode to 'visible'.");
else message=i18n("%1 sets the channel mode to 'visible'.").arg(sourceNick);
}
modeS->setDown(plus);
if(plus) modeP->setDown(false);
break;
//case 'r': break;
case 't':
if(plus)
{
if(fromMe) message=i18n("You switch on 'topic protection'.");
else message=i18n("%1 switches on 'topic protection'.").arg(sourceNick);
}
else
{
if(fromMe) message=i18n("You switch off 'topic protection'.");
else message=i18n("%1 switches off 'topic protection'.").arg(sourceNick);
}
modeT->setDown(plus);
break;
case 'k':
if(plus)
{
if(fromMe) message=i18n("You set the channel key to '%1'.").arg(parameter);
else message=i18n("%1 sets the channel key to '%2'.").arg(sourceNick).arg(parameter);
}
else
{
if(fromMe) message=i18n("You remove the channel key.");
else message=i18n("%1 removes the channel key.").arg(sourceNick);
}
modeK->setDown(plus);
break;
case 'l':
if(plus)
{
if(fromMe) message=i18n("You set the channel limit to %1 nicks.").arg(parameter);
else message=i18n("%1 sets the channel limit to %2 nicks.").arg(sourceNick).arg(parameter);
}
else
{
if(fromMe) message=i18n("You remove the channel limit.");
else message=i18n("%1 removes the channel limit.").arg(sourceNick);
}
modeL->setDown(plus);
if(plus) limit->setText(parameter);
else limit->clear();
break;
case 'b':
if(plus)
{
if(fromMe) message=i18n("You set a ban on %1.").arg(parameter);
else message=i18n("%1 sets a ban on %2.").arg(sourceNick).arg(parameter);
}
else
{
if(fromMe) message=i18n("You remove the ban on %1.").arg(parameter);
else message=i18n("%1 removes the ban on %2.").arg(sourceNick).arg(parameter);
}
break;
case 'e':
if(plus)
{
if(fromMe) message=i18n("You set a ban exception on %1.").arg(parameter);
else message=i18n("%1 sets a ban exception on %2.").arg(sourceNick).arg(parameter);
}
else
{
if(fromMe) message=i18n("You remove the ban exception on %1.").arg(parameter);
else message=i18n("%1 removes the ban exception on %2.").arg(sourceNick).arg(parameter);
}
break;
case 'I':
if(plus)
{
if(fromMe) message=i18n("You set invitation mask %1.").arg(parameter);
else message=i18n("%1 sets invitation mask %2.").arg(sourceNick).arg(parameter);
}
else
{
if(fromMe) message=i18n("You remove the invitation mask %1.").arg(parameter);
else message=i18n("%1 removes the invitation mask %2.").arg(sourceNick).arg(parameter);
}
break;
default:
if(plus)
{
if(fromMe) message=i18n("You set channel mode +%1").arg(mode);
else message=i18n("%1 sets channel mode +%2").arg(sourceNick).arg(mode);
}
else
{
if (fromMe) message=i18n("You set channel mode -%1").arg(mode);
else message= i18n("%1 sets channel mode -%2").arg(sourceNick).arg(mode);
}
}
// check if this nick's anyOp-status has changed and adjust ops accordingly
if(parameterChannelNick)
{
if(wasAnyOp && (!parameterChannelNick->isAnyTypeOfOp()))
adjustOps(-1);
else if((!wasAnyOp) && parameterChannelNick->isAnyTypeOfOp())
adjustOps(1);
}
if(!message.isEmpty() && !Preferences::useLiteralModes())
{
appendCommandMessage(i18n("Mode"),message);
}
updateModeWidgets(mode,plus,parameter);
}
void Channel::clearModeList()
{
TQString k;
// Keep channel password in the backing store, for rejoins.
for (TQStringList::const_iterator it = m_modeList.begin(); it != m_modeList.end(); ++it)
{
if ((*it)[0] == 'k') k = (*it);
}
m_modeList.clear();
if (!k.isEmpty()) m_modeList << k;
modeT->setOn(0);
modeT->setDown(0);
modeN->setOn(0);
modeN->setDown(0);
modeS->setOn(0);
modeS->setDown(0);
modeI->setOn(0);
modeI->setDown(0);
modeP->setOn(0);
modeP->setDown(0);
modeM->setOn(0);
modeM->setDown(0);
modeK->setOn(0);
modeK->setDown(0);
modeL->setOn(0);
modeL->setDown(0);
limit->clear();
emit modesChanged();
}
void Channel::updateModeWidgets(char mode, bool plus, const TQString &parameter)
{
ModeButton* widget=0;
if(mode=='t') widget=modeT;
else if(mode=='n') widget=modeN;
else if(mode=='s') widget=modeS;
else if(mode=='i') widget=modeI;
else if(mode=='p') widget=modeP;
else if(mode=='m') widget=modeM;
else if(mode=='k') widget=modeK;
else if(mode=='l')
{
widget=modeL;
if(plus) limit->setText(parameter);
else limit->clear();
}
if(widget) widget->setOn(plus);
if(plus)
{
m_modeList.append(TQString(mode + parameter));
}
else
{
TQStringList removable = m_modeList.grep(TQRegExp(TQString("^%1.*").arg(mode)));
for(TQStringList::iterator it = removable.begin(); it != removable.end(); ++it)
{
m_modeList.remove(m_modeList.find((*it)));
}
}
emit modesChanged();
}
void Channel::updateQuickButtons(const TQStringList &newButtonList)
{
// remove quick buttons from memory and GUI
while(buttonList.count())
{
QuickButton* button=buttonList.at(0);
// remove tool tips as well
TQToolTip::remove(button);
buttonList.remove(button);
delete button;
}
if(buttonsGrid)delete buttonsGrid;
// the grid that holds the quick action buttons
buttonsGrid = new TQGrid(2, nickListButtons);
// add new quick buttons
for(unsigned int index=0;index<newButtonList.count();index++)
{
// generate empty buttons first, text will be added later
QuickButton* quickButton = new QuickButton(TQString(), TQString(), buttonsGrid);
buttonList.append(quickButton);
connect(quickButton, TQT_SIGNAL(clicked(const TQString &)), this, TQT_SLOT(quickButtonClicked(const TQString &)));
// Get the button definition
TQString buttonText=newButtonList[index];
// Extract button label
TQString buttonLabel=buttonText.section(',',0,0);
// Extract button definition
buttonText=buttonText.section(',',1);
quickButton->setText(buttonLabel);
quickButton->setDefinition(buttonText);
// Add tool tips
TQString toolTip=buttonText.replace("&","&amp;").
replace("<","&lt;").
replace(">","&gt;");
TQToolTip::add(quickButton,toolTip);
quickButton->show();
} // for
// set hide() or show() on grid
showQuickButtons(Preferences::showQuickButtons());
}
void Channel::showQuickButtons(bool show)
{
// TQt does not redraw the buttons properly when they are not on screen
// while getting hidden, so we remember the "soon to be" state here.
if(isHidden() || !buttonsGrid)
{
quickButtonsChanged=true;
quickButtonsState=show;
}
else
{
if(show)
buttonsGrid->show();
else
buttonsGrid->hide();
}
}
void Channel::showModeButtons(bool show)
{
// TQt does not redraw the buttons properly when they are not on screen
// while getting hidden, so we remember the "soon to be" state here.
if(isHidden())
{
modeButtonsChanged=true;
modeButtonsState=show;
}
else
{
if(show)
{
topicSplitterHidden = false;
modeBox->show();
modeBox->parentWidget()->show();
}
else
{
modeBox->hide();
if(topicLine->isHidden())
{
topicSplitterHidden = true;
modeBox->parentWidget()->hide();
}
}
}
}
void Channel::indicateAway(bool show)
{
// TQt does not redraw the label properly when they are not on screen
// while getting hidden, so we remember the "soon to be" state here.
if(isHidden())
{
awayChanged=true;
awayState=show;
}
else
{
if(show)
awayLabel->show();
else
awayLabel->hide();
}
}
void Channel::showEvent(TQShowEvent*)
{
// If the show quick/mode button settings have changed, apply the changes now
if(quickButtonsChanged)
{
quickButtonsChanged=false;
showQuickButtons(quickButtonsState);
}
if(modeButtonsChanged)
{
modeButtonsChanged=false;
showModeButtons(modeButtonsState);
}
if(awayChanged)
{
awayChanged=false;
indicateAway(awayState);
}
syncSplitters();
}
void Channel::syncSplitters()
{
TQValueList<int> vertSizes = Preferences::topicSplitterSizes();
TQValueList<int> horizSizes = Preferences::channelSplitterSizes();
if (vertSizes.isEmpty())
{
vertSizes << m_topicButton->height() << (height() - m_topicButton->height());
Preferences::setTopicSplitterSizes(vertSizes);
}
if (horizSizes.isEmpty())
{
int listWidth = nicknameListView->columnWidth(0) + nicknameListView->columnWidth(1);
horizSizes << (width() - listWidth) << listWidth;
Preferences::setChannelSplitterSizes(horizSizes);
}
m_vertSplitter->setSizes(vertSizes);
m_horizSplitter->setSizes(horizSizes);
splittersInitialized = true;
}
void Channel::updateAppearance()
{
TQColor fg,bg,abg;
if(Preferences::inputFieldsBackgroundColor())
{
fg=Preferences::color(Preferences::ChannelMessage);
bg=Preferences::color(Preferences::TextViewBackground);
abg=Preferences::color(Preferences::AlternateBackground);
}
else
{
fg=colorGroup().foreground();
bg=colorGroup().base();
abg=TDEGlobalSettings::alternateBackgroundColor();
}
channelInput->unsetPalette();
channelInput->setPaletteForegroundColor(fg);
channelInput->setPaletteBackgroundColor(bg);
limit->unsetPalette();
limit->setPaletteForegroundColor(fg);
limit->setPaletteBackgroundColor(bg);
getTextView()->unsetPalette();
if(Preferences::showBackgroundImage())
{
getTextView()->setViewBackground(Preferences::color(Preferences::TextViewBackground),
Preferences::backgroundImage());
}
else
{
getTextView()->setViewBackground(Preferences::color(Preferences::TextViewBackground),
TQString());
}
if (Preferences::customTextFont())
{
getTextView()->setFont(Preferences::textFont());
topicLine->setFont(Preferences::textFont());
channelInput->setFont(Preferences::textFont());
nicknameCombobox->setFont(Preferences::textFont());
limit->setFont(Preferences::textFont());
}
else
{
getTextView()->setFont(TDEGlobalSettings::generalFont());
topicLine->setFont(TDEGlobalSettings::generalFont());
channelInput->setFont(TDEGlobalSettings::generalFont());
nicknameCombobox->setFont(TDEGlobalSettings::generalFont());
limit->setFont(TDEGlobalSettings::generalFont());
}
nicknameListView->resort();
nicknameListView->unsetPalette();
nicknameListView->setPaletteForegroundColor(fg);
nicknameListView->setPaletteBackgroundColor(bg);
nicknameListView->setAlternateBackground(abg);
if (Preferences::customListFont())
nicknameListView->setFont(Preferences::listFont());
else
nicknameListView->setFont(TDEGlobalSettings::generalFont());
nicknameListView->refresh();
showModeButtons(Preferences::showModeButtons());
showNicknameList(Preferences::showNickList());
showNicknameBox(Preferences::showNicknameBox());
showTopic(Preferences::showTopic());
setAutoUserhost(Preferences::autoUserhost());
updateQuickButtons(Preferences::quickButtonList());
ChatWindow::updateAppearance();
}
void Channel::nicknameComboboxChanged()
{
TQString newNick=nicknameCombobox->currentText();
oldNick=m_server->getNickname();
if(oldNick!=newNick)
{
nicknameCombobox->setCurrentText(oldNick);
changeNickname(newNick);
// return focus to input line
channelInput->setFocus();
}
}
void Channel::changeNickname(const TQString& newNickname)
{
if (!newNickname.isEmpty())
m_server->queue("NICK "+newNickname);
}
void Channel::resetNickList()
{
nicknameListView->setUpdatesEnabled(false);
purgeNicks();
}
void Channel::addPendingNickList(const TQStringList& pendingChannelNickList)
{
if(pendingChannelNickList.isEmpty())
return;
if (!m_processingTimer)
{
m_processingTimer = new TQTimer(this);
connect(m_processingTimer, TQT_SIGNAL(timeout()), this, TQT_SLOT(processPendingNicks()));
}
m_pendingChannelNickLists.append(pendingChannelNickList);
if (!m_processingTimer->isActive())
m_processingTimer->start(0);
}
void Channel::childAdjustFocus()
{
channelInput->setFocus();
refreshModeButtons();
}
void Channel::refreshModeButtons()
{
bool enable = true;
if(getOwnChannelNick())
{
enable=getOwnChannelNick()->isAnyTypeOfOp();
} // if not channel nick, then enable is true - fall back to assuming they are op
//don't disable the mode buttons since you can't then tell if they are enabled or not.
//needs to be fixed somehow
/* modeT->setEnabled(enable);
modeN->setEnabled(enable);
modeS->setEnabled(enable);
modeI->setEnabled(enable);
modeP->setEnabled(enable);
modeM->setEnabled(enable);
modeK->setEnabled(enable);
modeL->setEnabled(enable);*/
limit->setEnabled(enable);
// Tooltips for the ModeButtons
TQString opOnly;
if(!enable) opOnly = i18n("You have to be an operator to change this.");
TQToolTip::add(modeT, i18n("Topic can be changed by channel operator only. %1").arg(opOnly));
TQToolTip::add(modeN, i18n("No messages to channel from clients on the outside. %1").arg(opOnly));
TQToolTip::add(modeS, i18n("Secret channel. %1").arg(opOnly));
TQToolTip::add(modeI, i18n("Invite only channel. %1").arg(opOnly));
TQToolTip::add(modeP, i18n("Private channel. %1").arg(opOnly));
TQToolTip::add(modeM, i18n("Moderated channel. %1").arg(opOnly));
TQToolTip::add(modeK, i18n("Protect channel with a password."));
TQToolTip::add(modeL, i18n("Set user limit to channel."));
}
void Channel::cycleChannel()
{
closeYourself();
m_server->sendJoinCommand(getName(), getPassword());
}
void Channel::autoUserhost()
{
if(Preferences::autoUserhost() && !Preferences::autoWhoContinuousEnabled())
{
int limit = 5;
TQString nickString;
TQPtrList<Nick> nickList = getNickList();
TQPtrListIterator<Nick> it(nickList);
Nick* nick;
while((nick = it.current()) != 0)
{
if(nick->getChannelNick()->getHostmask().isEmpty())
{
if(limit--) nickString = nickString + nick->getChannelNick()->getNickname() + ' ';
else break;
}
++it;
}
if(!nickString.isEmpty()) m_server->requestUserhost(nickString);
}
}
void Channel::setAutoUserhost(bool state)
{
if(state)
{
// we can't have automatic resizing with three columns; the hostname column is too wide
nicknameListView->setHScrollBarMode(TQScrollView::Auto);
// restart userhost timer
userhostTimer.start(10000);
// if the column was actually gone (just to be sure) ...
if(nicknameListView->columns()==2)
{
// re-add the hostmask column
nicknameListView->addColumn(TQString());
nicknameListView->setColumnWidthMode(2,TDEListView::Maximum);
nicknameListView->setResizeMode(TQListView::NoColumn);
// re-add already known hostmasks
TQListViewItem* item=nicknameListView->itemAtIndex(0);
while(item)
{
Nick* lookNick=getNickByName(item->text(1));
if(lookNick) item->setText(2,lookNick->getChannelNick()->getHostmask());
item=item->itemBelow();
}
}
}
else
{
userhostTimer.stop();
if(nicknameListView->columns()==3) nicknameListView->removeColumn(2);
nicknameListView->setHScrollBarMode(TQScrollView::AlwaysOff);
// make the nick column resize itself automatically to prevent horizontal scrollbar
nicknameListView->setResizeMode(TQListView::LastColumn);
}
}
void Channel::scheduleAutoWho() // slot
{
if(m_whoTimer.isActive())
m_whoTimer.stop();
if(Preferences::autoWhoContinuousEnabled())
m_whoTimer.start(Preferences::autoWhoContinuousInterval() * 1000, true);
}
void Channel::autoWho()
{
// don't use auto /WHO when the number of nicks is too large, or get banned.
if((nicks > Preferences::autoWhoNicksLimit()) ||
m_server->getInputFilter()->isWhoRequestUnderProcess(getName()))
{
scheduleAutoWho();
return;
}
m_server->requestWho(getName());
}
void Channel::fadeActivity()
{
TQPtrListIterator<Nick> it( nicknameList );
Nick *nick;
while ( (nick = it.current()) != 0 ) {
++it;
nick->getChannelNick()->lessActive();
}
}
TQString Channel::getTextInLine()
{
return channelInput->text();
}
bool Channel::canBeFrontView()
{
return true;
}
bool Channel::searchView()
{
return true;
}
void Channel::appendInputText(const TQString& s, bool fromCursor)
{
if(!fromCursor)
{
channelInput->append(s);
}
else
{
int para = 0, index = 0;
channelInput->getCursorPosition(&para, &index);
channelInput->insertAt(s, para, index);
channelInput->setCursorPosition(para, index + s.length());
}
}
bool Channel::closeYourself(bool confirm)
{
int result=KMessageBox::Continue;
if (confirm)
result = KMessageBox::warningContinueCancel(this, i18n("Do you want to leave %1?").arg(getName()),
i18n("Leave Channel"), i18n("Leave"), "QuitChannelTab");
if (result==KMessageBox::Continue)
{
m_server->closeChannel(getName());
m_server->removeChannel(this);
Preferences::setSpellChecking(channelInput->checkSpellingEnabled());
deleteLater();
return true;
}
return false;
}
void Channel::serverOnline(bool online)
{
setActive(online);
}
//Used to disable functions when not connected, does not necessarily mean the server is offline
void Channel::setActive(bool active)
{
if (active)
{
getTextView()->setNickAndChannelContextMenusEnabled(true);
nicknameCombobox->setEnabled(true);
}
else
{
purgeNicks();
getTextView()->setNickAndChannelContextMenusEnabled(false);
nicknameCombobox->setEnabled(false);
topicLine->setText(TQString());
clearModeList();
clearBanList();
}
}
void Channel::showTopic(bool show)
{
if(show)
{
topicSplitterHidden = false;
topicLine->show();
m_topicButton->show();
topicLine->parentWidget()->show();
}
else
{
topicLine->hide();
m_topicButton->hide();
if(modeBox->isHidden())
{
topicSplitterHidden = true;
topicLine->parentWidget()->hide();
}
}
}
void Channel::processPendingNicks()
{
TQString nickname = m_pendingChannelNickLists.first()[m_currentIndex];
bool admin = false;
bool owner = false;
bool op = false;
bool halfop = false;
bool voice = false;
// Remove possible mode characters from nickname and store the resulting mode
m_server->mangleNicknameWithModes(nickname, admin, owner, op, halfop, voice);
// TODO: make these an enumeration in TDEApplication or somewhere, we can use them as well
unsigned int mode = (admin ? 16 : 0) +
(owner ? 8 : 0) +
(op ? 4 : 0) +
(halfop ? 2 : 0) +
(voice ? 1 : 0);
// Check if nick is already in the nicklist
if (!getNickByName(nickname))
{
ChannelNickPtr nick = m_server->addNickToJoinedChannelsList(getName(), nickname);
Q_ASSERT(nick);
nick->setMode(mode);
fastAddNickname(nick);
if (nick->isAdmin() || nick->isOwner() || nick->isOp() || nick->isHalfOp())
m_opsToAdd++;
m_currentIndex++;
}
else
{
m_pendingChannelNickLists.first().pop_front();
}
if (m_pendingChannelNickLists.first().count() <= m_currentIndex)
{
adjustNicks(m_pendingChannelNickLists.first().count());
adjustOps(m_opsToAdd);
m_pendingChannelNickLists.pop_front();
m_currentIndex = 0;
m_opsToAdd = 0;
}
if (m_pendingChannelNickLists.isEmpty())
{
m_processingTimer->stop();
nicknameListView->sort();
sortNickList();
nicknameListView->setUpdatesEnabled(true);
nicknameListView->triggerUpdate();
}
}
void Channel::setChannelEncoding(const TQString& encoding) // virtual
{
Preferences::setChannelEncoding(m_server->getDisplayName(), getName(), encoding);
}
TQString Channel::getChannelEncoding() // virtual
{
return Preferences::channelEncoding(m_server->getDisplayName(), getName());
}
TQString Channel::getChannelEncodingDefaultDesc() // virtual
{
return i18n("Identity Default ( %1 )").arg(getServer()->getIdentity()->getCodecName());
}
void Channel::showNicknameBox(bool show)
{
if(show)
{
nicknameCombobox->show();
}
else
{
nicknameCombobox->hide();
}
}
void Channel::showNicknameList(bool show)
{
if (show)
{
channelSplitterHidden = false;
nickListButtons->show();
}
else
{
channelSplitterHidden = true;
nickListButtons->hide();
}
}
void Channel::requestNickListSort()
{
if(!m_delayedSortTimer)
{
m_delayedSortTimer = new TQTimer(this);
connect(m_delayedSortTimer, TQT_SIGNAL(timeout()), this, TQT_SLOT(sortNickList()));
}
if(!m_delayedSortTimer->isActive())
{
m_delayedSortTimer->start(1000, true);
}
}
void Channel::sortNickList()
{
nicknameList.sort();
nicknameListView->resort();
if(m_delayedSortTimer)
{
m_delayedSortTimer->stop();
}
}
void Channel::setIdentity(const IdentityPtr identity)
{
if (identity)
{
nicknameCombobox->clear();
nicknameCombobox->insertStringList(identity->getNicknameList());
}
}
bool Channel::eventFilter(TQObject* watched, TQEvent* e)
{
if((TQT_BASE_OBJECT(watched) == TQT_BASE_OBJECT(nicknameListView)) && (e->type() == TQEvent::Resize) && splittersInitialized && isShown())
{
if (!topicSplitterHidden && !channelSplitterHidden)
{
Preferences::setChannelSplitterSizes(m_horizSplitter->sizes());
Preferences::setTopicSplitterSizes(m_vertSplitter->sizes());
}
if (!topicSplitterHidden && channelSplitterHidden)
{
Preferences::setTopicSplitterSizes(m_vertSplitter->sizes());
}
if (!channelSplitterHidden && topicSplitterHidden)
{
Preferences::setChannelSplitterSizes(m_horizSplitter->sizes());
}
}
return ChatWindow::eventFilter(watched, e);
}
void Channel::addBan(const TQString& ban)
{
for ( TQStringList::Iterator it = m_BanList.begin(); it != m_BanList.end(); ++it )
{
if ((*it).section(' ', 0, 0) == ban.section(' ', 0, 0))
{
// Ban is already in list.
it = m_BanList.remove(it);
emit banRemoved(ban.section(' ', 0, 0));
}
}
m_BanList.prepend(ban);
emit banAdded(ban);
}
void Channel::removeBan(const TQString& ban)
{
for ( TQStringList::Iterator it = m_BanList.begin(); it != m_BanList.end(); ++it )
{
if ((*it).section(' ', 0, 0) == ban)
{
it = m_BanList.remove(it);
emit banRemoved(ban);
}
}
}
void Channel::clearBanList()
{
m_BanList.clear();
emit banListCleared();
}
void Channel::append(const TQString& nickname, const TQString& message)
{
if(nickname != getServer()->getNickname()) {
Nick* nick = getNickByName(nickname);
if(nick) {
nick->getChannelNick()->setTimeStamp(TQDateTime::currentDateTime().toTime_t());
}
}
ChatWindow::append(nickname, message);
nickActive(nickname);
}
void Channel::appendAction(const TQString& nickname, const TQString& message)
{
if(nickname != getServer()->getNickname()) {
Nick* nick = getNickByName(nickname);
if(nick) {
nick->getChannelNick()->setTimeStamp(TQDateTime::currentDateTime().toTime_t());
}
}
ChatWindow::appendAction(nickname, message);
nickActive(nickname);
}
void Channel::nickActive(const TQString& nickname) //FIXME reported to crash, can't reproduce
{
ChannelNickPtr nick=getChannelNick(nickname);
//XXX Would be nice to know why it can be null here...
if (nick) {
nick->moreActive();
sortNickList(); //FIXME: no need to completely resort, we can just see if this particular nick should move
}
}
//
// NickList
//
NickList::NickList() : TQPtrList<Nick>()
{
m_compareMethod = NickList::AlphaNumeric;
}
void NickList::setCompareMethod(CompareMethod method)
{
m_compareMethod = method;
}
//doesn't the following somehow duplicate NickListViewItem::compare()?
int NickList::compareItems(TQPtrCollection::Item item1, TQPtrCollection::Item item2)
{
if(m_compareMethod == NickList::TimeStamp) {
int returnValue = static_cast<Nick*>(item2)->getChannelNick()->timeStamp() - static_cast<Nick*>(item1)->getChannelNick()->timeStamp();
if(returnValue == 0) {
returnValue = TQString::compare(static_cast<Nick*>(item1)->getChannelNick()->loweredNickname(),
static_cast<Nick*>(item2)->getChannelNick()->loweredNickname());
}
return returnValue;
}
return TQString::compare(static_cast<Nick*>(item1)->getChannelNick()->loweredNickname(),
static_cast<Nick*>(item2)->getChannelNick()->loweredNickname());
}
TQString NickList::completeNick(const TQString& pattern, bool& complete, TQStringList& found,
bool skipNonAlfaNum, bool caseSensitive)
{
found.clear();
TQString prefix = "^";
TQString newNick;
TQString prefixCharacter = Preferences::prefixCharacter();
NickList foundNicks;
foundNicks.setCompareMethod(NickList::TimeStamp);
if((pattern.find(TQRegExp("^(\\d|\\w)")) != -1) && skipNonAlfaNum)
{
prefix = "^([^\\d\\w]|[\\_]){0,}";
}
TQRegExp regexp(prefix + TQRegExp::escape(pattern));
regexp.setCaseSensitive(caseSensitive);
TQPtrListIterator<Nick> it(*this);
while(it.current() != 0)
{
newNick = it.current()->getChannelNick()->getNickname();
if(!prefix.isEmpty() && newNick.contains(prefixCharacter))
{
newNick = newNick.section( prefixCharacter,1 );
}
if(newNick.find(regexp) != -1)
{
foundNicks.append(it.current());
}
++it;
}
foundNicks.sort();
TQPtrListIterator<Nick> it2(foundNicks);
while(it2.current() != 0)
{
found.append(it2.current()->getChannelNick()->getNickname());
++it2;
}
if(found.count() > 1)
{
bool ok = true;
unsigned int patternLength = pattern.length();
TQString firstNick = found[0];
unsigned int firstNickLength = firstNick.length();
unsigned int foundCount = found.count();
while(ok && ((patternLength) < firstNickLength))
{
++patternLength;
TQStringList tmp = found.grep(firstNick.left(patternLength), caseSensitive);
if(tmp.count() != foundCount)
{
ok = false;
--patternLength;
}
}
complete = false;
return firstNick.left(patternLength);
}
else if(found.count() == 1)
{
complete = true;
return found[0];
}
return TQString();
}
bool NickList::containsNick(const TQString& nickname)
{
TQPtrListIterator<Nick> it(*this);
while (it.current() != 0)
{
if (it.current()->getChannelNick()->getNickname()==nickname)
return true;
++it;
}
return false;
}
#include "channel.moc"
// kate: space-indent on; tab-width 4; indent-width 4; mixed-indent off; replace-tabs on;
// vim: set et sw=4 ts=4 cino=l1,cs,U1: