/***************************************************************************
- * Copyright (C) 2005-2010 by the Quassel Project *
+ * Copyright (C) 2005-2018 by the Quassel Project *
* devel@quassel-irc.org *
* *
* This program is free software; you can redistribute it and/or modify *
* You should have received a copy of the GNU General Public License *
* along with this program; if not, write to the *
* Free Software Foundation, Inc., *
- * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. *
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
***************************************************************************/
#include "ctcpparser.h"
+#include "corenetworkconfig.h"
#include "coresession.h"
#include "ctcpevent.h"
#include "messageevent.h"
+#include "coreuserinputhandler.h"
const QByteArray XDELIM = "\001";
CtcpParser::CtcpParser(CoreSession *coreSession, QObject *parent)
- : QObject(parent),
+ : QObject(parent),
_coreSession(coreSession)
{
- QByteArray MQUOTE = QByteArray("\020");
- _ctcpMDequoteHash[MQUOTE + '0'] = QByteArray(1, '\000');
- _ctcpMDequoteHash[MQUOTE + 'n'] = QByteArray(1, '\n');
- _ctcpMDequoteHash[MQUOTE + 'r'] = QByteArray(1, '\r');
- _ctcpMDequoteHash[MQUOTE + MQUOTE] = MQUOTE;
+ QByteArray MQUOTE = QByteArray("\020");
+ _ctcpMDequoteHash[MQUOTE + '0'] = QByteArray(1, '\000');
+ _ctcpMDequoteHash[MQUOTE + 'n'] = QByteArray(1, '\n');
+ _ctcpMDequoteHash[MQUOTE + 'r'] = QByteArray(1, '\r');
+ _ctcpMDequoteHash[MQUOTE + MQUOTE] = MQUOTE;
- QByteArray XQUOTE = QByteArray("\134");
- _ctcpXDelimDequoteHash[XQUOTE + XQUOTE] = XQUOTE;
- _ctcpXDelimDequoteHash[XQUOTE + QByteArray("a")] = XDELIM;
+ setStandardCtcp(_coreSession->networkConfig()->standardCtcp());
- connect(this, SIGNAL(newEvent(Event *)), _coreSession->eventManager(), SLOT(postEvent(Event *)));
+ connect(_coreSession->networkConfig(), SIGNAL(standardCtcpSet(bool)), this, SLOT(setStandardCtcp(bool)));
+ connect(this, SIGNAL(newEvent(Event *)), _coreSession->eventManager(), SLOT(postEvent(Event *)));
}
+
+void CtcpParser::setStandardCtcp(bool enabled)
+{
+ QByteArray XQUOTE = QByteArray("\134");
+ if (enabled)
+ _ctcpXDelimDequoteHash[XQUOTE + XQUOTE] = XQUOTE;
+ else
+ _ctcpXDelimDequoteHash.remove(XQUOTE + XQUOTE);
+ _ctcpXDelimDequoteHash[XQUOTE + QByteArray("a")] = XDELIM;
+}
+
+
void CtcpParser::displayMsg(NetworkEvent *event, Message::Type msgType, const QString &msg, const QString &sender,
- const QString &target, Message::Flags msgFlags) {
- if(event->testFlag(EventManager::Silent))
- return;
+ const QString &target, Message::Flags msgFlags)
+{
+ if (event->testFlag(EventManager::Silent))
+ return;
- MessageEvent *msgEvent = new MessageEvent(msgType, event->network(), msg, sender, target, msgFlags);
- msgEvent->setTimestamp(event->timestamp());
+ MessageEvent *msgEvent = new MessageEvent(msgType, event->network(), msg, sender, target, msgFlags);
+ msgEvent->setTimestamp(event->timestamp());
- emit newEvent(msgEvent);
+ emit newEvent(msgEvent);
}
-QByteArray CtcpParser::lowLevelQuote(const QByteArray &message) {
- QByteArray quotedMessage = message;
- QHash<QByteArray, QByteArray> quoteHash = _ctcpMDequoteHash;
- QByteArray MQUOTE = QByteArray("\020");
- quoteHash.remove(MQUOTE + MQUOTE);
- quotedMessage.replace(MQUOTE, MQUOTE + MQUOTE);
+QByteArray CtcpParser::lowLevelQuote(const QByteArray &message)
+{
+ QByteArray quotedMessage = message;
+
+ QHash<QByteArray, QByteArray> quoteHash = _ctcpMDequoteHash;
+ QByteArray MQUOTE = QByteArray("\020");
+ quoteHash.remove(MQUOTE + MQUOTE);
+ quotedMessage.replace(MQUOTE, MQUOTE + MQUOTE);
- QHash<QByteArray, QByteArray>::const_iterator quoteIter = quoteHash.constBegin();
- while(quoteIter != quoteHash.constEnd()) {
- quotedMessage.replace(quoteIter.value(), quoteIter.key());
- quoteIter++;
- }
- return quotedMessage;
+ QHash<QByteArray, QByteArray>::const_iterator quoteIter = quoteHash.constBegin();
+ while (quoteIter != quoteHash.constEnd()) {
+ quotedMessage.replace(quoteIter.value(), quoteIter.key());
+ ++quoteIter;
+ }
+ return quotedMessage;
}
-QByteArray CtcpParser::lowLevelDequote(const QByteArray &message) {
- QByteArray dequotedMessage;
- QByteArray messagepart;
- QHash<QByteArray, QByteArray>::iterator ctcpquote;
-
- // copy dequote Message
- for(int i = 0; i < message.size(); i++) {
- messagepart = message.mid(i,1);
- if(i+1 < message.size()) {
- for(ctcpquote = _ctcpMDequoteHash.begin(); ctcpquote != _ctcpMDequoteHash.end(); ++ctcpquote) {
- if(message.mid(i,2) == ctcpquote.key()) {
- messagepart = ctcpquote.value();
- ++i;
- break;
+
+QByteArray CtcpParser::lowLevelDequote(const QByteArray &message)
+{
+ QByteArray dequotedMessage;
+ QByteArray messagepart;
+ QHash<QByteArray, QByteArray>::iterator ctcpquote;
+
+ // copy dequote Message
+ for (int i = 0; i < message.size(); i++) {
+ messagepart = message.mid(i, 1);
+ if (i+1 < message.size()) {
+ for (ctcpquote = _ctcpMDequoteHash.begin(); ctcpquote != _ctcpMDequoteHash.end(); ++ctcpquote) {
+ if (message.mid(i, 2) == ctcpquote.key()) {
+ messagepart = ctcpquote.value();
+ ++i;
+ break;
+ }
+ }
}
- }
+ dequotedMessage += messagepart;
}
- dequotedMessage += messagepart;
- }
- return dequotedMessage;
+ return dequotedMessage;
}
-QByteArray CtcpParser::xdelimQuote(const QByteArray &message) {
- QByteArray quotedMessage = message;
- QHash<QByteArray, QByteArray>::const_iterator quoteIter = _ctcpXDelimDequoteHash.constBegin();
- while(quoteIter != _ctcpXDelimDequoteHash.constEnd()) {
- quotedMessage.replace(quoteIter.value(), quoteIter.key());
- quoteIter++;
- }
- return quotedMessage;
+
+QByteArray CtcpParser::xdelimQuote(const QByteArray &message)
+{
+ QByteArray quotedMessage = message;
+ QHash<QByteArray, QByteArray>::const_iterator quoteIter = _ctcpXDelimDequoteHash.constBegin();
+ while (quoteIter != _ctcpXDelimDequoteHash.constEnd()) {
+ quotedMessage.replace(quoteIter.value(), quoteIter.key());
+ ++quoteIter;
+ }
+ return quotedMessage;
}
-QByteArray CtcpParser::xdelimDequote(const QByteArray &message) {
- QByteArray dequotedMessage;
- QByteArray messagepart;
- QHash<QByteArray, QByteArray>::iterator xdelimquote;
-
- for(int i = 0; i < message.size(); i++) {
- messagepart = message.mid(i,1);
- if(i+1 < message.size()) {
- for(xdelimquote = _ctcpXDelimDequoteHash.begin(); xdelimquote != _ctcpXDelimDequoteHash.end(); ++xdelimquote) {
- if(message.mid(i,2) == xdelimquote.key()) {
- messagepart = xdelimquote.value();
- i++;
- break;
+
+QByteArray CtcpParser::xdelimDequote(const QByteArray &message)
+{
+ QByteArray dequotedMessage;
+ QByteArray messagepart;
+ QHash<QByteArray, QByteArray>::iterator xdelimquote;
+
+ for (int i = 0; i < message.size(); i++) {
+ messagepart = message.mid(i, 1);
+ if (i+1 < message.size()) {
+ for (xdelimquote = _ctcpXDelimDequoteHash.begin(); xdelimquote != _ctcpXDelimDequoteHash.end(); ++xdelimquote) {
+ if (message.mid(i, 2) == xdelimquote.key()) {
+ messagepart = xdelimquote.value();
+ i++;
+ break;
+ }
+ }
}
- }
+ dequotedMessage += messagepart;
}
- dequotedMessage += messagepart;
- }
- return dequotedMessage;
+ return dequotedMessage;
}
-void CtcpParser::processIrcEventRawNotice(IrcEventRawMessage *event) {
- parse(event, Message::Notice);
+
+void CtcpParser::processIrcEventRawNotice(IrcEventRawMessage *event)
+{
+ parse(event, Message::Notice);
}
-void CtcpParser::processIrcEventRawPrivmsg(IrcEventRawMessage *event) {
- parse(event, Message::Plain);
+
+void CtcpParser::processIrcEventRawPrivmsg(IrcEventRawMessage *event)
+{
+ parse(event, Message::Plain);
}
-void CtcpParser::parse(IrcEventRawMessage *e, Message::Type messagetype) {
- QByteArray ctcp;
- //lowlevel message dequote
- QByteArray dequotedMessage = lowLevelDequote(e->rawMessage());
+void CtcpParser::parse(IrcEventRawMessage *e, Message::Type messagetype)
+{
+ //lowlevel message dequote
+ QByteArray dequotedMessage = lowLevelDequote(e->rawMessage());
+
+ CtcpEvent::CtcpType ctcptype = e->type() == EventManager::IrcEventRawNotice
+ ? CtcpEvent::Reply
+ : CtcpEvent::Query;
- CtcpEvent::CtcpType ctcptype = e->type() == EventManager::IrcEventRawNotice
- ? CtcpEvent::Reply
- : CtcpEvent::Query;
+ Message::Flags flags = (ctcptype == CtcpEvent::Reply && !e->network()->isChannelName(e->target()))
+ ? Message::Redirected
+ : Message::None;
- Message::Flags flags = (ctcptype == CtcpEvent::Reply && !e->network()->isChannelName(e->target()))
- ? Message::Redirected
- : Message::None;
+ bool isStatusMsg = false;
+
+ // First remove all statusmsg prefix characters that are not also channel prefix characters.
+ while (e->network()->isStatusMsg(e->target()) && !e->network()->isChannelName(e->target())) {
+ isStatusMsg = true;
+ e->setTarget(e->target().remove(0, 1));
+ }
- QList<CtcpEvent *> ctcpEvents;
- QUuid uuid; // needed to group all replies together
+ // Then continue removing statusmsg characters as long as removing the character will still result in a
+ // valid channel name. This prevents removing the channel prefix character if said character is in the
+ // overlap between the statusmsg characters and the channel prefix characters.
+ while (e->network()->isStatusMsg(e->target()) && e->network()->isChannelName(e->target().remove(0, 1))) {
+ isStatusMsg = true;
+ e->setTarget(e->target().remove(0, 1));
+ }
- // extract tagged / extended data
- int xdelimPos = -1;
- int xdelimEndPos = -1;
- int spacePos = -1;
- while((xdelimPos = dequotedMessage.indexOf(XDELIM)) != -1) {
- if(xdelimPos > 0)
- displayMsg(e, messagetype, targetDecode(e, dequotedMessage.left(xdelimPos)), e->prefix(), e->target(), flags);
+ // If any statusmsg characters were removed, Flag the message as a StatusMsg.
+ if (isStatusMsg) {
+ flags |= Message::StatusMsg;
+ }
- xdelimEndPos = dequotedMessage.indexOf(XDELIM, xdelimPos + 1);
- if(xdelimEndPos == -1) {
- // no matching end delimiter found... treat rest of the message as ctcp
- xdelimEndPos = dequotedMessage.count();
+ // For self-messages, pass the flag on to the message, too
+ if (e->testFlag(EventManager::Self)) {
+ flags |= Message::Self;
}
- ctcp = xdelimDequote(dequotedMessage.mid(xdelimPos + 1, xdelimEndPos - xdelimPos - 1));
- dequotedMessage = dequotedMessage.mid(xdelimEndPos + 1);
- //dispatch the ctcp command
- QString ctcpcmd = targetDecode(e, ctcp.left(spacePos));
- QString ctcpparam = targetDecode(e, ctcp.mid(spacePos + 1));
+ if (coreSession()->networkConfig()->standardCtcp())
+ parseStandard(e, messagetype, dequotedMessage, ctcptype, flags);
+ else
+ parseSimple(e, messagetype, dequotedMessage, ctcptype, flags);
+}
+
- spacePos = ctcp.indexOf(' ');
- if(spacePos != -1) {
- ctcpcmd = targetDecode(e, ctcp.left(spacePos));
- ctcpparam = targetDecode(e, ctcp.mid(spacePos + 1));
+// only accept CTCPs in their simplest form, i.e. one ctcp, from start to
+// end, no text around it; not as per the 'specs', but makes people happier
+void CtcpParser::parseSimple(IrcEventRawMessage *e, Message::Type messagetype, QByteArray dequotedMessage, CtcpEvent::CtcpType ctcptype, Message::Flags flags)
+{
+ if (dequotedMessage.count(XDELIM) != 2 || dequotedMessage[0] != '\001' || dequotedMessage[dequotedMessage.count() -1] != '\001') {
+ displayMsg(e, messagetype, targetDecode(e, dequotedMessage), e->prefix(), e->target(), flags);
} else {
- ctcpcmd = targetDecode(e, ctcp);
- ctcpparam = QString();
+ int spacePos;
+ QString ctcpcmd, ctcpparam;
+
+ QByteArray ctcp = xdelimDequote(dequotedMessage.mid(1, dequotedMessage.count() - 2));
+ spacePos = ctcp.indexOf(' ');
+ if (spacePos != -1) {
+ ctcpcmd = targetDecode(e, ctcp.left(spacePos));
+ ctcpparam = targetDecode(e, ctcp.mid(spacePos + 1));
+ } else {
+ ctcpcmd = targetDecode(e, ctcp);
+ ctcpparam = QString();
+ }
+ ctcpcmd = ctcpcmd.toUpper();
+
+ // we don't want to block /me messages by the CTCP ignore list
+ if (ctcpcmd == QLatin1String("ACTION") || !coreSession()->ignoreListManager()->ctcpMatch(e->prefix(), e->network()->networkName(), ctcpcmd)) {
+ QUuid uuid = QUuid::createUuid();
+ _replies.insert(uuid, CtcpReply(coreNetwork(e), nickFromMask(e->prefix())));
+ CtcpEvent *event = new CtcpEvent(EventManager::CtcpEvent, e->network(), e->prefix(), e->target(),
+ ctcptype, ctcpcmd, ctcpparam, e->timestamp(), uuid);
+ emit newEvent(event);
+ CtcpEvent *flushEvent = new CtcpEvent(EventManager::CtcpEventFlush, e->network(), e->prefix(), e->target(),
+ ctcptype, "INVALID", QString(), e->timestamp(), uuid);
+ emit newEvent(flushEvent);
+ }
}
+}
- ctcpcmd = ctcpcmd.toUpper();
- // we don't want to block /me messages by the CTCP ignore list
- if(ctcpcmd == QLatin1String("ACTION") || !coreSession()->ignoreListManager()->ctcpMatch(e->prefix(), e->network()->networkName(), ctcpcmd)) {
- if(uuid.isNull())
- uuid = QUuid::createUuid();
+void CtcpParser::parseStandard(IrcEventRawMessage *e, Message::Type messagetype, QByteArray dequotedMessage, CtcpEvent::CtcpType ctcptype, Message::Flags flags)
+{
+ QByteArray ctcp;
+
+ QList<CtcpEvent *> ctcpEvents;
+ QUuid uuid; // needed to group all replies together
+
+ // extract tagged / extended data
+ int xdelimPos = -1;
+ int xdelimEndPos = -1;
+ int spacePos = -1;
+ while ((xdelimPos = dequotedMessage.indexOf(XDELIM)) != -1) {
+ if (xdelimPos > 0)
+ displayMsg(e, messagetype, targetDecode(e, dequotedMessage.left(xdelimPos)), e->prefix(), e->target(), flags);
+
+ xdelimEndPos = dequotedMessage.indexOf(XDELIM, xdelimPos + 1);
+ if (xdelimEndPos == -1) {
+ // no matching end delimiter found... treat rest of the message as ctcp
+ xdelimEndPos = dequotedMessage.count();
+ }
+ ctcp = xdelimDequote(dequotedMessage.mid(xdelimPos + 1, xdelimEndPos - xdelimPos - 1));
+ dequotedMessage = dequotedMessage.mid(xdelimEndPos + 1);
+
+ //dispatch the ctcp command
+ QString ctcpcmd = targetDecode(e, ctcp.left(spacePos));
+ QString ctcpparam = targetDecode(e, ctcp.mid(spacePos + 1));
+
+ spacePos = ctcp.indexOf(' ');
+ if (spacePos != -1) {
+ ctcpcmd = targetDecode(e, ctcp.left(spacePos));
+ ctcpparam = targetDecode(e, ctcp.mid(spacePos + 1));
+ }
+ else {
+ ctcpcmd = targetDecode(e, ctcp);
+ ctcpparam = QString();
+ }
+
+ ctcpcmd = ctcpcmd.toUpper();
+
+ // we don't want to block /me messages by the CTCP ignore list
+ if (ctcpcmd == QLatin1String("ACTION") || !coreSession()->ignoreListManager()->ctcpMatch(e->prefix(), e->network()->networkName(), ctcpcmd)) {
+ if (uuid.isNull())
+ uuid = QUuid::createUuid();
- CtcpEvent *event = new CtcpEvent(EventManager::CtcpEvent, e->network(), e->prefix(), e->target(),
- ctcptype, ctcpcmd, ctcpparam, e->timestamp(), uuid);
- ctcpEvents << event;
+ CtcpEvent *event = new CtcpEvent(EventManager::CtcpEvent, e->network(), e->prefix(), e->target(),
+ ctcptype, ctcpcmd, ctcpparam, e->timestamp(), uuid);
+ ctcpEvents << event;
+ }
}
- }
- if(!ctcpEvents.isEmpty()) {
- _replies.insert(uuid, CtcpReply(coreNetwork(e), nickFromMask(e->prefix())));
- CtcpEvent *flushEvent = new CtcpEvent(EventManager::CtcpEventFlush, e->network(), e->prefix(), e->target(),
- ctcptype, "INVALID", QString(), e->timestamp(), uuid);
- ctcpEvents << flushEvent;
- foreach(CtcpEvent *event, ctcpEvents) {
- emit newEvent(event);
+ if (!ctcpEvents.isEmpty()) {
+ _replies.insert(uuid, CtcpReply(coreNetwork(e), nickFromMask(e->prefix())));
+ CtcpEvent *flushEvent = new CtcpEvent(EventManager::CtcpEventFlush, e->network(), e->prefix(), e->target(),
+ ctcptype, "INVALID", QString(), e->timestamp(), uuid);
+ ctcpEvents << flushEvent;
+ foreach(CtcpEvent *event, ctcpEvents) {
+ emit newEvent(event);
+ }
}
- }
- if(!dequotedMessage.isEmpty())
- displayMsg(e, messagetype, targetDecode(e, dequotedMessage), e->prefix(), e->target(), flags);
+ if (!dequotedMessage.isEmpty())
+ displayMsg(e, messagetype, targetDecode(e, dequotedMessage), e->prefix(), e->target(), flags);
}
-void CtcpParser::sendCtcpEvent(CtcpEvent *e) {
- CoreNetwork *net = coreNetwork(e);
- if(e->type() == EventManager::CtcpEvent) {
- QByteArray quotedReply;
- QString bufname = nickFromMask(e->prefix());
- if(e->ctcpType() == CtcpEvent::Query && !e->reply().isNull()) {
- if(_replies.contains(e->uuid()))
- _replies[e->uuid()].replies << lowLevelQuote(pack(net->serverEncode(e->ctcpCmd()),
- net->userEncode(bufname, e->reply())));
- else
- // reply not caused by a request processed in here, so send it off immediately
- reply(net, bufname, e->ctcpCmd(), e->reply());
+
+void CtcpParser::sendCtcpEvent(CtcpEvent *e)
+{
+ CoreNetwork *net = coreNetwork(e);
+ if (e->type() == EventManager::CtcpEvent) {
+ QByteArray quotedReply;
+ QString bufname = nickFromMask(e->prefix());
+ if (e->ctcpType() == CtcpEvent::Query && !e->reply().isNull()) {
+ if (_replies.contains(e->uuid()))
+ _replies[e->uuid()].replies << lowLevelQuote(pack(net->serverEncode(e->ctcpCmd()),
+ net->userEncode(bufname, e->reply())));
+ else
+ // reply not caused by a request processed in here, so send it off immediately
+ reply(net, bufname, e->ctcpCmd(), e->reply());
+ }
+ }
+ else if (e->type() == EventManager::CtcpEventFlush && _replies.contains(e->uuid())) {
+ CtcpReply reply = _replies.take(e->uuid());
+ if (reply.replies.count())
+ packedReply(net, reply.bufferName, reply.replies);
}
- } else if(e->type() == EventManager::CtcpEventFlush && _replies.contains(e->uuid())) {
- CtcpReply reply = _replies.take(e->uuid());
- if(reply.replies.count())
- packedReply(net, reply.bufferName, reply.replies);
- }
}
-QByteArray CtcpParser::pack(const QByteArray &ctcpTag, const QByteArray &message) {
- if(message.isEmpty())
- return XDELIM + ctcpTag + XDELIM;
- return XDELIM + ctcpTag + ' ' + xdelimQuote(message) + XDELIM;
+QByteArray CtcpParser::pack(const QByteArray &ctcpTag, const QByteArray &message)
+{
+ if (message.isEmpty())
+ return XDELIM + ctcpTag + XDELIM;
+
+ return XDELIM + ctcpTag + ' ' + xdelimQuote(message) + XDELIM;
}
-void CtcpParser::query(CoreNetwork *net, const QString &bufname, const QString &ctcpTag, const QString &message) {
- QList<QByteArray> params;
- params << net->serverEncode(bufname) << lowLevelQuote(pack(net->serverEncode(ctcpTag), net->userEncode(bufname, message)));
- net->putCmd("PRIVMSG", params);
+
+void CtcpParser::query(CoreNetwork *net, const QString &bufname, const QString &ctcpTag, const QString &message)
+{
+ QString cmd("PRIVMSG");
+
+ std::function<QList<QByteArray>(QString &)> cmdGenerator = [&] (QString &splitMsg) -> QList<QByteArray> {
+ return QList<QByteArray>() << net->serverEncode(bufname) << lowLevelQuote(pack(net->serverEncode(ctcpTag), net->userEncode(bufname, splitMsg)));
+ };
+
+ net->putCmd(cmd, net->splitMessage(cmd, message, cmdGenerator));
}
-void CtcpParser::reply(CoreNetwork *net, const QString &bufname, const QString &ctcpTag, const QString &message) {
- QList<QByteArray> params;
- params << net->serverEncode(bufname) << lowLevelQuote(pack(net->serverEncode(ctcpTag), net->userEncode(bufname, message)));
- net->putCmd("NOTICE", params);
+
+void CtcpParser::reply(CoreNetwork *net, const QString &bufname, const QString &ctcpTag, const QString &message)
+{
+ QList<QByteArray> params;
+ params << net->serverEncode(bufname) << lowLevelQuote(pack(net->serverEncode(ctcpTag), net->userEncode(bufname, message)));
+ net->putCmd("NOTICE", params);
}
-void CtcpParser::packedReply(CoreNetwork *net, const QString &bufname, const QList<QByteArray> &replies) {
- QList<QByteArray> params;
-
- int answerSize = 0;
- for(int i = 0; i < replies.count(); i++) {
- answerSize += replies.at(i).size();
- }
-
- QByteArray quotedReply(answerSize, 0);
- int nextPos = 0;
- QByteArray &reply = quotedReply;
- for(int i = 0; i < replies.count(); i++) {
- reply = replies.at(i);
- quotedReply.replace(nextPos, reply.size(), reply);
- nextPos += reply.size();
- }
-
- params << net->serverEncode(bufname) << quotedReply;
- // FIXME user proper event
- net->putCmd("NOTICE", params);
+
+void CtcpParser::packedReply(CoreNetwork *net, const QString &bufname, const QList<QByteArray> &replies)
+{
+ QList<QByteArray> params;
+
+ int answerSize = 0;
+ for (int i = 0; i < replies.count(); i++) {
+ answerSize += replies.at(i).size();
+ }
+
+ QByteArray quotedReply;
+ quotedReply.reserve(answerSize);
+ for (int i = 0; i < replies.count(); i++) {
+ quotedReply.append(replies.at(i));
+ }
+
+ params << net->serverEncode(bufname) << quotedReply;
+ // FIXME user proper event
+ net->putCmd("NOTICE", params);
}