summaryrefslogtreecommitdiffstats
path: root/ksirc/chanparser.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'ksirc/chanparser.cpp')
-rw-r--r--ksirc/chanparser.cpp1047
1 files changed, 1047 insertions, 0 deletions
diff --git a/ksirc/chanparser.cpp b/ksirc/chanparser.cpp
new file mode 100644
index 00000000..8429a6f5
--- /dev/null
+++ b/ksirc/chanparser.cpp
@@ -0,0 +1,1047 @@
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+#endif
+
+#include "alistbox.h"
+#include "chanparser.h"
+#include "ksopts.h"
+#include "control_message.h"
+#include "ssfeprompt.h"
+#include "toplevel.h"
+#include "ksircprocess.h"
+#include "ksview.h"
+
+#include <stdio.h>
+
+#include <qregexp.h>
+#include <qapplication.h>
+#include <klocale.h>
+#include <kmessagebox.h>
+#include <kdebug.h>
+#include <qptrlist.h>
+
+// Static parser table is "initialized"
+QDict<parseFunc> ChannelParser::parserTable;
+
+
+ChannelParser::ChannelParser(KSircTopLevel *_top)
+{
+ top = _top;
+
+ /*
+ * Initial helper variables
+ */
+ prompt_active = false;
+ current_item = -1;
+ top_item = 0;
+
+ if(parserTable.isEmpty() == TRUE){
+ parserTable.setAutoDelete(TRUE);
+ parserTable.insert("`l`", new parseFunc(&ChannelParser::parseSSFEClear));
+ parserTable.insert("`s`", new parseFunc(&ChannelParser::parseSSFEStatus));
+ parserTable.insert("`i`", new parseFunc(&ChannelParser::parseSSFEInit));
+ parserTable.insert("`t`", new parseFunc(&ChannelParser::parseSSFEMsg));
+ parserTable.insert("`o`", new parseFunc(&ChannelParser::parseSSFEOut));
+ parserTable.insert("`p`", new parseFunc(&ChannelParser::parseSSFEPrompt));
+ parserTable.insert("`P`", new parseFunc(&ChannelParser::parseSSFEPrompt));
+ parserTable.insert("`R`", new parseFunc(&ChannelParser::parseSSFEReconnect));
+ // The rest are *** info message
+ parserTable.insert("***", new parseFunc(&ChannelParser::parseINFOInfo));
+ parserTable.insert("*E*", new parseFunc(&ChannelParser::parseINFOError));
+ parserTable.insert("*!*", new parseFunc(&ChannelParser::parseINFONicks)); // Normal
+ parserTable.insert("*C*", new parseFunc(&ChannelParser::parseINFONicks)); // 1st line
+ parserTable.insert("*c*", new parseFunc(&ChannelParser::parseINFONicks)); // Last line
+ parserTable.insert("*#*", new parseFunc(&ChannelParser::parseINFONicks)); // Non enhanced
+ parserTable.insert("*$*", new parseFunc(&ChannelParser::parseINFONicks)); // Enhanced turned off
+ parserTable.insert("*>*", new parseFunc(&ChannelParser::parseINFOJoin));
+ parserTable.insert("*<*", new parseFunc(&ChannelParser::parseINFOPart));
+ parserTable.insert("*N*", new parseFunc(&ChannelParser::parseINFOChangeNick));
+ parserTable.insert("*+*", new parseFunc(&ChannelParser::parseINFOMode));
+ parserTable.insert("*T*", new parseFunc(&ChannelParser::parseINFOTopic));
+ // End of info message
+ parserTable.insert("* ", new parseFunc(&ChannelParser::parseCTCPAction));
+ }
+
+}
+
+parseResult *ChannelParser::parse(QString string)
+{
+ // for older Qts
+
+ parseFunc *pf;
+ if(string.length() < 3){
+ return new parseError(string, QString("Dumb string, too short"));
+ }
+
+ /**
+ * Start pre-parsing the strings to make them fit the 3 character
+ * parser codes, etc
+ */
+
+ /*
+ * SSFE control messages are too long, so we convert the
+ * messges into a 3 character code, `#ssfe#\S becomes `\S`
+ */
+
+ if ((string[0] == '`') & (string.length() > 7))
+ {
+ QString prefix = "`"+string[7]+"`";
+ string = prefix + QString(" ") + string.mid(8).stripWhiteSpace();
+ }
+ else if((string[0] == '*') && (string[1] == ' ')) {
+ string.insert(1, ' ');
+ }
+ // Pre-parsing is now complete
+
+ pf = parserTable[string.mid(0, 3)];
+ if(pf != 0x0){
+ parseResult *result = (this->*(pf->parser))(string);
+
+ if (result)
+ return result;
+ }
+ else
+ {
+ // debug("No handler for: %s", string.data());
+ }
+
+ // Little bit of past parsing to catch one we've missed
+ if((string[0] == '*') && (string[2] == '*')) {
+ string.remove(0, 3);
+ return new parseSucc(string, ksopts->infoColor, "user|servinfo");
+ }
+ // If it's unkown we just fall out of the function
+ return 0;
+}
+
+parseResult * ChannelParser::parseSSFEClear(QString string)
+{
+// top->mainw->clear();
+ top->clearWindow();
+// top->mainw->repaint(TRUE);
+ string.truncate(0);
+ return new parseSucc(QString::null); // Null string, don't display anything
+}
+
+parseResult * ChannelParser::parseSSFEStatus(QString string)
+{
+ string.remove(0, 12); // strip off the first 12 characters "<junk> [sirc] "
+ if(string.length() == 0)
+ return new parseError("", i18n("Unable to parse status string"));
+
+ //kdDebug(5008) << "String: " << string << endl;
+ QRegExp rx("(\\S+).*\\(*([+-]*[+-\\w\\d]*)\\)*.*on (\\S+) \\((\\S+)\\)");
+ if(rx.search(string) == -1){
+ return new parseError("", i18n("Unable to parse status (no known format) string"));
+ }
+
+ QString nick = rx.cap(1);
+ QString modes = rx.cap(2);
+ QString chan = rx.cap(3);
+ QString chanmode = rx.cap(4);
+
+ /*
+ * fix up modes which may have gotten the (away) part
+ */
+ if(modes.contains("away")){
+ modes = "";
+ }
+
+ bool away = false;
+ if(string.contains("(away)"))
+ away = true;
+
+ if(away){
+ chan.prepend(i18n("Away-"));
+ }
+
+ nickListItem *nickItem = top->nicks->item( top->nicks->findNick( nick ) );
+ if ( nickItem ) {
+ if(nickItem->away() != away){
+ nickItem->setAway( away );
+ top->nicks->viewport()->repaint( top->nicks->itemRect( nickItem ), false );
+ }
+ nickItem->forceColour(&ksopts->ownNickColor);
+ }
+
+ top->ksircProcess()->setNick(nick);
+ if (chanmode.findRev("t") != -1)
+ top->channelButtons->setProtectMode(true);
+ else top->channelButtons->setProtectMode(false);
+ if (chanmode.findRev("m") != -1)
+ top->channelButtons->setModerateMode(true);
+ else top->channelButtons->setModerateMode(false);
+ if (chanmode.findRev("n") != -1)
+ top->channelButtons->setNooutsideMode(true);
+ else top->channelButtons->setNooutsideMode(false);
+ if (chanmode.findRev("i") != -1)
+ top->channelButtons->setMenuItemMode(0, true);
+ else top->channelButtons->setMenuItemMode(0, false);
+ if (chanmode.findRev("s") != -1)
+ top->channelButtons->setMenuItemMode(3, true);
+ else top->channelButtons->setMenuItemMode(3, false);
+
+ if (modes.findRev("i") != -1)
+ top->channelButtons->setMenuItemMode(4, true);
+ else top->channelButtons->setMenuItemMode(4, false);
+ if (modes.findRev("w") != -1)
+ top->channelButtons->setMenuItemMode(5, true);
+ else top->channelButtons->setMenuItemMode(5, false);
+ if (modes.findRev("s") != -1)
+ top->channelButtons->setMenuItemMode(6, true);
+ else top->channelButtons->setMenuItemMode(6, false);
+
+ QString status_line = QString("%1 (%2) %3 (%4) ").arg(chan).arg(chanmode).arg(nick).arg(modes);
+
+ /*
+ * Go srearching for key and limit messages
+ */
+ QRegExp rxKey("<key: (\\S+)>");
+ if(rxKey.search(string) >= 0){
+ top->channelButtons->setMenuItemMode(2, true);
+ status_line += QString("<key: %1>").arg(rxKey.cap(1));
+ }
+ else {
+ top->channelButtons->setMenuItemMode(2, false);
+ }
+
+ QRegExp rxLimit("<limit: (\\S+)>");
+ if(rxLimit.search(string) >= 0){
+ top->channelButtons->setMenuItemMode(1, true);
+ status_line += QString("<limit: %1>").arg(rxLimit.cap(1));
+ }
+ else {
+ top->channelButtons->setMenuItemMode(1, false);
+ }
+
+ if(ksopts->displayTopic){
+ if(top->topic().length() > 0)
+ status_line += "T: " + top->topic();
+ else
+ status_line += "T: " + i18n("<No Topic Set>");
+ }
+
+ if(top->caption != status_line){
+ if(nick[0] == '@' || (nick[0] == '*' && nick[1] == '@')) {
+ // If we're an op,,
+ // update the nicks popup menu
+ top->channelButtons->setButtonsEnabled(true); // set the buttons enabled if were an op
+ top->opami = TRUE;
+ } // opami = true sets us to an op
+ else {
+ top->channelButtons->setButtonsEnabled(false); // set the buttons enabled if were an op
+ top->opami = FALSE;
+ } // FALSE, were not an ops
+ top->UserUpdateMenu(); // update the menu
+ top->setCaption(status_line);
+ top->setIconText(status_line);
+ if(top->ticker) {
+ top->ticker->setCaption(status_line);
+ }
+ top->caption = status_line; // Make copy so we're not
+ // constantly changing the title bar
+ }
+ return new parseSucc(QString::null); // Null string, don't display anything
+}
+
+parseResult * ChannelParser::parseSSFEInit(QString)
+{
+ return new parseSucc(QString::null); // Null string, don't display anything
+}
+
+parseResult * ChannelParser::parseSSFEOut(QString)
+{
+ return new parseSucc(QString::null); // Null string, don't display anything
+}
+
+parseResult * ChannelParser::parseSSFEMsg(QString string)
+{
+
+ if(string.length() > 100)
+ return new parseError(QString::null, i18n("String length for nick is greater than 100 characters. This is unacceptably long."));
+
+ int l = string.length();
+ if (l <= 0)
+ return new parseError(string, i18n("String not long enough"));
+
+ return new parseSucc(QString::null); // Null string, don't display anything
+}
+
+
+parseResult * ChannelParser::parseSSFEPrompt(QString string)
+{
+
+ if(prompt_active == FALSE){
+ QString prompt, caption;
+ ssfePrompt *sp;
+
+ // Flush the screen.
+ // First remove the prompt message from the Buffer.
+ // (it's garunteed to be the first one)
+ top->LineBuffer.remove( *top->LineBuffer.begin() );
+ top->Buffer = FALSE;
+ top->sirc_receive(QString(""));
+
+ // "'[pP]' " gives 4 spaces
+ if(string.length() < 5)
+ caption = i18n("");
+ else
+ caption = string.mid(3);
+ prompt_active = TRUE;
+
+ // If we use this, then it blows up
+ // if we haven't popped up on the remote display yet.
+
+ KSirc::TextParagIterator it = top->mainw->firstParag();
+ QString last;
+ while(it.atEnd() == FALSE) {
+ last = it.plainText();
+ ++it;
+ }
+
+ if(last[0] == '['){ /* strip time stamp */
+ prompt = last.mid(last.find(' '));
+ }
+ else {
+ prompt = last;
+ }
+
+ sp = new ssfePrompt(prompt, 0);
+ sp->setCaption(caption);
+ if(string[1] == 'P')
+ sp->setPassword(TRUE);
+ sp->exec();
+ // cerr << "Entered: " << sp->text() << endl;
+ prompt = sp->text();
+ prompt += "\n";
+ emit top->outputUnicodeLine(prompt);
+ delete sp;
+ prompt_active = FALSE;
+ }
+
+ return new parseSucc(QString::null); // Null string, don't display anything
+}
+
+parseResult * ChannelParser::parseSSFEReconnect(QString)
+{
+
+ if(top->channelInfo().channel()[0] == '#' ||
+ top->channelInfo().channel()[0] == '&'){
+ QString str = "/join " + QString(top->channelInfo().channel()) + "\n";
+ emit top->outputUnicodeLine(str);
+ }
+
+ return new parseSucc(QString::null); // Null string, don't display anything
+}
+
+parseResult * ChannelParser::parseINFOInfo(QString string)
+{
+ string.remove(0, 3); // takes off the junk
+
+ return new parseSucc(string, ksopts->infoColor, "user|servinfo"); // Null string, don't display anything
+}
+
+parseResult * ChannelParser::parseINFOError(QString string)
+{
+ string.remove(0, 3); // strip the junk
+
+ return new parseSucc(string,ksopts->errorColor, "user|error"); // Null string, don't display anything
+}
+
+parseResult * ChannelParser::parseINFONicks(QString in_string)
+{
+ QString string = in_string;
+ QString channel_name;
+ bool clear_box = FALSE;
+
+ // Check to see if it's a continued line
+ if(string[1] == 'C'){
+ string[1] = '!';
+ clear_box = TRUE;
+ }
+ if(string[1] == '#'){
+ string[1] = '!';
+ clear_box = FALSE;
+ }
+ else if(string[1] == 'c'){
+ if(current_item > 0)
+ top->nicks->setCurrentItem(current_item);
+ top->nicks->setTopItem(top_item);
+ top->nicks->repaint(TRUE);
+ return new parseSucc(QString::null); // Parsing ok, don't print anything though
+ }
+ else if(string[1] == '$'){
+ top->nicks->clearAdvOps();
+ //kdDebug(5008) << "Turning off advanced ops" << endl;
+ return new parseSucc(QString::null); // Parsing ok, don't print anything though
+ }
+
+
+ // Get the channel name portion of the string
+ // Search for the first space, since : can be embeded into channel names.
+ //count = sscanf(string, "*!* Users on %100[^ ]", channelInfo().channel());
+ // *!* Users on #TEST: boren asj asj_
+ QRegExp rx("\\*\\S\\* Users on (\\S+): (.+)");
+ if(rx.search(string) == -1){
+ return new parseError(string, i18n("Could not find channel name"));
+ }
+ channel_name = rx.cap(1);
+
+
+ if (channel_name.lower() != top->channelInfo().channel().lower()){
+ string.remove(0,3);
+ return new parseSucc(string,ksopts->infoColor,"user|misc4");
+ }
+
+ if(clear_box == TRUE){
+ current_item = top->nicks->currentItem();
+ top_item = top->nicks->topItem();
+ top->nicks->clear();
+ }
+
+ //int start = string.find(": ", 0, FALSE); // Find start of nicks
+ //if (start < 0)
+ // return new parseError(string, i18n("Could not find start of nicks"));
+ //
+ //place_holder = new char[string.length()];
+ //strcpy(place_holder, string.ascii()+start+2);
+ //nick = strtok(place_holder, " ");
+ // while(nick != 0x0){ // While there's nick to go...
+ QStringList nicks = QStringList::split(QRegExp("\\s+"), rx.cap(2));
+
+ for ( QStringList::Iterator it = nicks.begin(); it != nicks.end(); ++it ) {
+ QString nick = *it;
+ nickListItem *irc = new nickListItem();
+
+ bool done = FALSE;
+ uint i;
+
+ for(i = 0; i < nick.length();i++){
+ switch(nick[0].unicode()){
+ case '@':
+ irc->setOp(TRUE);
+ nick.remove(0,1);
+ break;
+ case '+':
+ irc->setVoice(TRUE);
+ nick.remove(0,1);
+ break;
+ case '#':
+ irc->setAway(TRUE);
+ nick.remove(0,1);
+ break;
+ case '*':
+ irc->setIrcOp(TRUE);
+ nick.remove(0,1);
+ break;
+ default:
+ done = TRUE;
+ }
+ if(done == TRUE)
+ break;
+ }
+
+ if(nick == top->ksircProcess()->getNick()){
+ irc->forceColour(&ksopts->ownNickColor);
+ }
+ irc->setText(nick);
+ top->nicks->inSort(irc);
+ }
+
+ return new parseSucc(QString::null); // Parsing ok, don't print anything though
+}
+
+parseResult * ChannelParser::parseINFOJoin(QString string)
+{
+ string.remove(0, 4); // strip *>* to save a few compares
+ // You have joined channel #Linux
+ QRegExp rx("You have joined channel (\\S+)");
+ if(rx.search(string) != -1){
+ //QString channel = rx.cap(1).lower();
+ QString channel = rx.cap(1);
+ //kdDebug(5008) << "Channel: " << channel << endl;
+
+ if(top->channelInfo().channel() != channel) {
+ KSircChannel ci(top->channelInfo().server(), channel);
+ kdDebug(5008) << "Warning: we got a channel join yet me don't belong to it!!! Assuming no key!" << endl;
+ kdDebug(5008) << "String was: " << string << endl;
+ kdDebug(5008) << "We think the channel is: " << channel << " we are: " << top->channelInfo().channel()<< endl;
+ emit top->open_toplevel(ci);
+ }
+ return new parseJoinPart(" " + string, ksopts->channelColor, "user|join");
+ }
+
+ // reef-diddy ([email protected]) has joined channel #aquaria
+ rx.setPattern("(\\S+) .+ has joined channel (\\S+)");
+ if(rx.search(string) != -1){
+ QString nick = rx.cap(1);
+ QString channel = rx.cap(2).lower();
+ //kdDebug(5008) << "Channel: " << channel << " nick: " << nick << endl;
+ if(top->channelInfo().channel().lower() != channel){
+ return new parseWrongChannel(" " + string, ksopts->errorColor, "user|join");
+ }
+ // nicks->insertItem(s3, 0); // add the sucker
+ top->nicks->inSort(nick);
+ top->addCompleteNick(nick);
+ highlightNick(string, nick);
+ return new parseJoinPart(" " + string, ksopts->channelColor, "user|join");
+ }
+
+ return 0; // ??
+}
+
+parseResult * ChannelParser::parseINFOPart(QString string)
+{
+
+ bool foundNick = false;
+ QString pixname = "user|kick";
+ QString nick("");
+
+ string.remove(0, 4); // clear junk
+
+ // Multiple type of parts, a signoff or a /part
+ // Each get's get nick in a diffrent localtion
+ // Set we search and find the nick and the remove it from the nick list
+ // 1. /quit, signoff, nick after "^Singoff: "
+ // 2. /part, leave the channek, nick after "has left \w+$"
+ // 3. /kick, kicked off the channel, nick after "kicked off \w+$"
+ //
+ // Signoff: looser
+ QRegExp rx("Signoff: (\\S+)");
+ if(rx.search(string) != -1) {
+ nick = rx.cap(1);
+ foundNick = true;
+ pixname = "user|X";
+ highlightNick(string, nick);
+ }
+ /*
+ * Check for "You" before everyone else or else the next
+ * case will match it
+ * You have left channel <channel>
+ */
+ rx.setPattern("You have left channel (\\S+)");
+ if((foundNick == false) && (rx.search(string) != -1)) {
+ QString channel = rx.cap(1);
+ if(top->channelInfo().channel().lower() == channel.lower()) {
+ QApplication::postEvent(top, new QCloseEvent());
+ // WE'RE DEAD
+ return new parseSucc(QString::null);
+ }
+ pixname = "user|part";
+ }
+ /*
+ * Same as above, check your own state first
+ * You have been kicked off channel <channel>
+ */
+ rx.setPattern("You have been kicked off channel (\\S+)");
+ if((foundNick == false) && (rx.search(string) != -1)) {
+ QString channel = rx.cap(1);
+ if(top->channelInfo().channel().lower() != channel.lower())
+ return new parseWrongChannel(string, ksopts->errorColor, "user|kick");
+ if (ksopts->autoRejoin == TRUE)
+ {
+ QString str = QString("/join %1\n").arg(top->channelInfo().channel());
+ emit top->outputUnicodeLine(str);
+ /* if(top->ticker)
+ top->ticker->show();
+ else*/
+ top->show();
+ }
+ else
+ {
+ if(top->KickWinOpen != false)
+ return new parseError(" " + string, i18n("Kick window open"));
+ top->KickWinOpen = true;
+ int result = KMessageBox::questionYesNo(top, string, i18n("You Have Been Kicked"), i18n("Rejoin"), i18n("Leave"));
+ if (result == KMessageBox::Yes)
+ {
+ QString str = QString("/join %1\n").arg(top->channelInfo().channel());
+ emit top->outputUnicodeLine(str);
+ /* if(top->ticker)
+ * top->ticker->show();
+ * else*/
+ top->show();
+ return new parseJoinPart(" " + string, ksopts->channelColor, "user|kick");
+ }
+ else
+ {
+ // WE'RE DEAD
+ QApplication::postEvent(top, new QCloseEvent());
+ top->KickWinOpen = false;
+ }
+ }
+ pixname = "user|kick";
+ }
+ /*
+ * <nick> has left channel <channel>
+ */
+ rx.setPattern("(\\S+) has left channel (\\S+)");
+ if((foundNick == false) && (rx.search(string) != -1)) {
+ nick = rx.cap(1);
+ QString channel = rx.cap(2);
+// kdDebug(5008) << "Nick: " << nick << " Channel: " << channel << " top: " << top->channelInfo().channel() << endl;
+ if(top->channelInfo().channel().lower() == channel.lower()) {
+ foundNick = true;
+ }
+ else{
+ return new parseWrongChannel(QString::null);
+ }
+ pixname = "user|part";
+ highlightNick(string, nick);
+ }
+ /*
+ * "<nick> has been kicked off channel <channel>"
+ */
+ rx.setPattern("(\\S+) has been kicked off channel (\\S+)");
+ if((foundNick == false) && (rx.search(string) != -1)) {
+ nick = rx.cap(1);
+ QString channel = rx.cap(2);
+ if(top->channelInfo().channel().lower() == channel.lower()) {
+ foundNick = true;
+ } else {
+ return new parseWrongChannel(QString::null);
+ }
+ highlightNick(string, nick);
+ pixname = "user|kick";
+ }
+
+ if (foundNick) {
+
+ top->removeCompleteNick(nick);
+
+ int index = top->nicks->findNick(nick);
+ if(index >= 0){
+ top->nicks->removeItem(index);
+ return new parseJoinPart(" " + string, ksopts->channelColor, pixname);
+ }
+ else {
+ return new parseJoinPart(QString::null);
+ }
+ }
+ else {
+ return new parseError(" " + string, i18n("Failed to parse part/kick/leave/quit message"));
+ }
+
+ return 0;
+}
+
+parseResult * ChannelParser::parseINFOChangeNick(QString string)
+{
+ //char old_nick[101], new_nick[101];
+ QString old_nick, new_nick;
+
+ string.remove(0, 4); // Remove the leading *N* and space
+
+ /*
+ * *N* asj_ is now known as bleh
+ */
+ //kdDebug(5008) << "Nick change: " << string << endl;
+ QRegExp rx("(\\S+) is now known as (\\S+)");
+ if(rx.search(string) == -1){
+ if(string.contains("already taken")){
+ return new parseSucc(" " + string, ksopts->errorColor, "user|error");
+ }
+
+ return new parseError(i18n("Unable to parse: %1").arg(string), i18n("Unable to parse change nick code"));
+ }
+
+ old_nick = rx.cap(1);
+ new_nick = rx.cap(2);
+
+ // If we have a window open talking to the nick
+ // Change the nick to the new one.
+ if((top->channelInfo().channel()[0] != '#' || top->channelInfo().channel()[0] != '&') &&
+ (top->channelInfo().channel() == old_nick)){
+ if(!top->ksircProcess()->mrList()[new_nick.lower()]){
+ top->control_message(CHANGE_CHANNEL, new_nick.lower());
+ }
+ }
+
+ highlightNick(string, old_nick);
+ highlightNick(string, new_nick);
+
+ // search the list for the nick and remove it
+ // since the list is source we should do a binary search...
+ int found = top->nicks->findNick(old_nick);
+ if(found >= 0){ // If the nick's in the nick list, change it and display the change
+ // save current selection
+ int selection = top->nicks->currentItem();
+
+ // Get the old item, and create a new one
+ nickListItem *it = top->nicks->item(found);
+ nickListItem *irc = new nickListItem(*it);
+ irc->setText(new_nick);
+
+ top->nicks->removeItem(found); // remove old nick
+ top->nicks->inSort(irc);
+
+ top->changeCompleteNick(old_nick, new_nick);
+
+ top->nicks->setCurrentItem(selection);
+ top->nicks->repaint(TRUE);
+ // We're done, so let's finish up
+ return new parseSucc(" " + string, ksopts->channelColor, "user|join");
+ }
+ else {
+ if(top->channelInfo().channel() == new_nick ||
+ top->channelInfo().channel() == old_nick)
+ return new parseSucc(" " + string, ksopts->channelColor, "user|elipsis");
+ else
+ return new parseSucc(QString::null);
+
+ }
+
+ // warning("Toplevel-N: nick change search failed on %s", s3.data());
+ // return new parseSucc(QString::null);
+}
+
+class mode_info {
+public:
+ mode_info(bool op, QChar mode, QString arg);
+
+ bool op() const;
+ QChar mode() const;
+ QString arg() const;
+
+private:
+ const bool m_op;
+ const QChar m_mode;
+ const QString m_arg;
+};
+
+mode_info::mode_info(bool op, QChar mode, QString arg) :
+ m_op(op),
+ m_mode(mode),
+ m_arg(arg)
+{
+}
+
+bool mode_info::op() const {
+ return m_op;
+}
+
+QChar mode_info::mode() const {
+ return m_mode;
+}
+
+QString mode_info::arg() const {
+ return m_arg;
+}
+
+
+parseResult * ChannelParser::parseINFOMode(QString string)
+{
+ // Basic idea here is simple, go through the mode change and
+ // assign each mode a + or a - and an argument or "" if there is
+ // none. After that each mode change it looked at to see if
+ // we should handle it in any special way.
+
+ // Strip off leading sirc info
+
+ string.remove(0, 4);
+
+
+ /*
+ * 1k is pretty safe since KProcess returns 1 k blocks, and lines don't get split between reads. This is emprical
+ */
+ QString modes, args, channel;
+ int found = 0;
+
+ if(string.find("for user") >= 0)
+ return new parseSucc(" " + string, ksopts->infoColor, "user|mode");
+
+ /*
+ * We need to 2 scanf's, one for the case of arguments, and one for no args.
+ */
+ QRegExp rx("Mode change \"(\\S+) *([^\"]*)\" on channel (\\S+)");
+ if(rx.search(string) >= 0){
+ modes = rx.cap(1);
+ args = rx.cap(2);
+ channel = rx.cap(3);
+ found = 1;
+ }
+
+ rx.setPattern("Mode for channel (\\S+) is \"([^\" ]+)\"");
+ if(found == 0 &&
+ rx.search(string) >= 0){
+ channel = rx.cap(1);
+ modes = rx.cap(2);
+ found = 1;
+ }
+
+ rx.setPattern("Your user mode is");
+ if(found == 0 &&
+ rx.search(string) >= 0){
+ /*
+ * Don't parse user mode requests
+ */
+ return new parseSucc(" " + string, ksopts->infoColor, "user|mode");
+ }
+
+
+ if(found == 0)
+ return new parseError(" Failed to parse mode change: " + string, QString::null);
+
+ /*
+ * op specifie if it's a + or -. tru is + false is -
+ */
+ bool op = true;
+ /*
+ * arglist is the list of argument
+ * we use the itirator to tstep through the list
+ * as need be
+ */
+ QStringList arglist = QStringList::split(" ", args);
+ QStringList::Iterator ai = arglist.begin();
+
+ /*
+ * the ptr list structure contains the parsed contents
+ */
+ QPtrList<const mode_info> pmList;
+ pmList.setAutoDelete(true);
+
+ for(uint pos = 0; pos < modes.length(); pos++){
+ switch(modes.at(pos).unicode()){
+ case '+':
+ op = true;
+ break;
+ case '-':
+ op = false;
+ break;
+ case 'l': // Chan limits
+ /*
+ * -l doesn't take any arguments, so just add the mode and break
+ * +l otoh does, so read the argument
+ */
+ if(op == false){
+ pmList.append(new mode_info(op, 'l', QString::null));
+ break;
+ }
+ case 'o': // Op, arg is the nick
+ case 'v': // Voice, arg is the nick
+ case 'b': // Ban, arg is mask banned
+ case 'k': // kcik, arg is nick
+ if(ai == NULL)
+ return new parseError(i18n("Unable to parse mode change: %1").arg(string), QString::null);
+ pmList.append(new mode_info(op, modes.at(pos), *ai));
+ ai++;
+ break;
+ case 'i': // Invite only
+ case 'n': // No message to chan
+ case 'p': // Private
+ case 'm': // Moderated
+ case 's': // Secret
+ case 't': // Topic setable by ops
+ case 'R': // (Dalnet) only registered may join
+ case 'r': // (Dalnet) only registered may join or something
+ /*
+ * Mode changes which don't take args
+ */
+ pmList.append(new mode_info(op, modes.at(pos), QString::null));
+ break;
+ default:
+ kdDebug(5008) << "Unknown mode change: " << modes.mid(pos, 1) << " Assume no args" << endl;
+ pmList.append(new mode_info(op, modes.at(pos), QString::null));
+ }
+ }
+ // We have the modes set in mode and arg, now we go though
+ // looking at each mode seeing if we should handle it.
+ bool mode_o_plus = false;
+ bool mode_o_minus = false;
+ bool mode_b_plus = false;
+ bool mode_b_minus = false;
+
+ QPtrListIterator<const mode_info> it(pmList);
+ const mode_info *mi;
+ while ( (mi = it.current()) != 0 ) {
+ ++it;
+ /*
+ * Look at the second character, it's uniq, check for +,- latter
+ */
+ if(mi->mode().unicode() == 'o'){
+ mode_o_plus = mi->op();
+ mode_o_minus = !mi->op();
+
+ if(top->ksircProcess()->getNick() == mi->arg())
+ top->channelButtons->setButtonsEnabled(mi->op());
+
+ if(mi->arg().length() == 0){
+ qWarning("Invalid nick in +/- o mode change");
+ continue;
+ }
+
+ int offset = top->nicks->findNick(mi->arg());
+ if(offset >= 0){
+ nickListItem *irc = new nickListItem();
+ *irc = *top->nicks->item(offset);
+ top->nicks->removeItem(offset); // remove old nick
+ irc->setOp(mi->op());
+ // add new nick in sorted pass,with colour
+ top->nicks->inSort(irc);
+ top->nicks->repaint(TRUE);
+ }
+ else{
+ kdDebug(5008) << "Toplevel+o: nick search failed on " << mi->arg() << endl;
+ }
+ }
+ else if(mi->mode() == 't'){
+ if(mi->op())
+ top->channelButtons->setProtectMode(true); // set on
+ else
+ top->channelButtons->setProtectMode(false); // set off
+ }
+ else if(mi->mode() == 'm'){
+ if(mi->op())
+ top->channelButtons->setModerateMode(true); // set on
+ else
+ top->channelButtons->setModerateMode(false); // set off
+ }
+ else if(mi->mode() == 'n'){
+ if(mi->op())
+ top->channelButtons->setNooutsideMode(true); // set on
+ else
+ top->channelButtons->setNooutsideMode(false); // set off
+ }
+ else if(mi->mode() == 'v'){
+ bool voice;
+ if(mi->op())
+ voice = TRUE;
+ else
+ voice = FALSE;
+
+ if(mi->arg().length() == 0){
+ qWarning("Invalid nick in +-v mode change");
+ continue;
+ }
+
+ int offset = top->nicks->findNick(mi->arg());
+ if(offset >= 0){
+ nickListItem *irc = new nickListItem();
+ *irc = *top->nicks->item(offset);
+ top->nicks->removeItem(offset); // remove old nick
+ irc->setVoice(voice) ;
+ // add new nick in sorted pass,with colour
+ top->nicks->inSort(irc);
+ top->nicks->repaint();
+ }
+ }
+ else if(mi->mode() == 'b'){
+ if(mi->op())
+ mode_b_plus = true;
+ else
+ mode_b_minus = true;
+ }
+ else if(mi->mode() == 'k'){
+ if(mi->op()){
+ if(mi->arg().length() == 0){
+ qWarning("Invalid +k mode set, no argument!");
+ continue;
+ }
+ top->m_channelInfo.setKey(mi->arg());
+ }
+ else {
+ top->m_channelInfo.setKey(""); /* no key set anymore */
+ }
+ }
+ else{
+ QChar c(mi->mode());
+ QString m(&c, 1);
+ QString o;
+ if(mi->op())
+ o = "+";
+ else
+ o = "-";
+ kdDebug(5008) << "Did not handle: " << o << m << " arg: " << mi->arg() << endl;
+ }
+ }
+ /*
+ * We're all done, so output the message and be done with it
+ */
+ QString pixname = "user|mode";
+
+ if(mode_o_plus)
+ pixname = "user|oplus";
+ else if(mode_o_minus)
+ pixname = "user|ominus";
+ else if(mode_b_plus)
+ pixname ="user|bplus";
+ else if(mode_b_minus)
+ pixname = "user|bminus";
+
+ return new parseSucc(" " + string, ksopts->infoColor, pixname);
+}
+
+parseResult * ChannelParser::parseCTCPAction(QString string)
+{
+
+ string.remove(0, 2); // * <something> use fancy * pixmap. Remove 2, leave one for space after te *
+ // why? looks cool for dorks
+ return new parseSucc(string, ksopts->textColor, "user|action");
+}
+
+parseResult * ChannelParser::parseINFOTopic(QString string)
+{
+
+ int found = 0;
+
+ string.remove(0, 4); // Remove the leading *T* and space
+
+ //kdDebug(5008) << "Topic parser: " << string << endl;
+
+ // Topic for #boo: this is a nice test
+ QRegExp rx( "Topic for (\\S+): (.*)" );
+ if(rx.search( string ) != -1){
+ QString channel = rx.cap(1);
+ QString topic = rx.cap(2);
+ topic.replace( QRegExp( "~~" ), "~" );
+
+ /*
+ * check where it's going.
+ * topic's maybe for other channels since they have no channel routing
+ * information, so route it to the right place if need be.
+ * If we're not on the channnel just fall through and display it
+ * on our channel, maybe the user asked for a topic of a different channel
+ */
+
+ if(channel.lower() != top->channelInfo().channel().lower()){
+ if(top->ksircProcess()->mrList()[channel.lower()]){
+ KSircTopLevel *t = dynamic_cast<KSircTopLevel *>(top->ksircProcess()->mrList()[channel.lower()]);
+ if(t)
+ t->setTopic(topic);
+ }
+ }
+ else {
+ //kdDebug(5008) << "New topic: " << topic << endl;
+ top->setTopic( topic );
+ }
+ found = 1;
+ }
+
+ rx.setPattern("(\\S+) has changed the topic on channel (\\S+) to (.+)");
+ if(found == 0 && rx.search(string) != -1){
+ QString nick = rx.cap(1);
+ QString channel = rx.cap(2);
+ //kdDebug(5008) << "Channel: " << channel << endl;
+ if(top->channelInfo().channel().lower() == channel.lower()){
+ QString topic = rx.cap(3);
+ //kdDebug(5008) << "Topic: " << topic << endl;
+ topic.replace(QRegExp("~~"), "~");
+ /*
+ * topic is in double quotes, so remove them
+ */
+ top->setTopic( topic.mid(1, topic.length()-2) );
+ QString cmd = "/eval &dostatus();\n";
+ top->sirc_write(cmd);
+ }
+ highlightNick(string, nick);
+ }
+ return new parseSucc(" " + string, ksopts->infoColor, "user|topic");
+}
+
+
+void ChannelParser::highlightNick(QString &string, QString &nick)
+{
+ QRegExp rx(QString("(^|\\s+)%1(\\s+|$)").arg(QRegExp::escape(nick)));
+ string.replace(rx, "\\1~n" + nick + "~n\\2");
+}
+