Fixing BR #39 (splitting long messages)
[quassel.git] / src / core / basichandler.cpp
1 /***************************************************************************
2  *   Copyright (C) 2005-08 by the Quassel Project                          *
3  *   devel@quassel-irc.org                                                 *
4  *                                                                         *
5  *   This program is free software; you can redistribute it and/or modify  *
6  *   it under the terms of the GNU General Public License as published by  *
7  *   the Free Software Foundation; either version 2 of the License, or     *
8  *   (at your option) version 3.                                           *
9  *                                                                         *
10  *   This program is distributed in the hope that it will be useful,       *
11  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        *
12  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         *
13  *   GNU General Public License for more details.                          *
14  *                                                                         *
15  *   You should have received a copy of the GNU General Public License     *
16  *   along with this program; if not, write to the                         *
17  *   Free Software Foundation, Inc.,                                       *
18  *   59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.             *
19  ***************************************************************************/
20 #include "basichandler.h"
21
22 #include <QMetaMethod>
23
24 #include "util.h"
25
26 BasicHandler::BasicHandler(NetworkConnection *parent)
27   : QObject(parent),
28     defaultHandler(-1),
29     _networkConnection(parent),
30     initDone(false)
31 {
32   connect(this, SIGNAL(displayMsg(Message::Type, BufferInfo::Type, QString, QString, QString, Message::Flags)),
33          networkConnection(), SIGNAL(displayMsg(Message::Type, BufferInfo::Type, QString, QString, QString, Message::Flags)));
34
35   connect(this, SIGNAL(putCmd(QString, const QList<QByteArray> &, const QByteArray &)),
36           networkConnection(), SLOT(putCmd(QString, const QList<QByteArray> &, const QByteArray &)));
37
38   connect(this, SIGNAL(putRawLine(const QByteArray &)),
39           networkConnection(), SLOT(putRawLine(const QByteArray &)));
40 }
41
42 QStringList BasicHandler::providesHandlers() {
43   return handlerHash().keys();
44 }
45
46 const QHash<QString, int> &BasicHandler::handlerHash() {
47   if(!initDone) {
48     for(int i = metaObject()->methodOffset(); i < metaObject()->methodCount(); i++) {
49       QString methodSignature(metaObject()->method(i).signature());
50       if(methodSignature.startsWith("defaultHandler")) {
51         defaultHandler = i;
52         continue;
53       }
54       
55       if(!methodSignature.startsWith("handle"))
56         continue;
57       
58       methodSignature = methodSignature.section('(',0,0);  // chop the attribute list
59       methodSignature = methodSignature.mid(6); // strip "handle"
60       _handlerHash[methodSignature] = i;
61     }
62     initDone = true;
63   }
64   return _handlerHash;
65 }
66
67 void BasicHandler::handle(const QString &member, QGenericArgument val0,
68                           QGenericArgument val1, QGenericArgument val2,
69                           QGenericArgument val3, QGenericArgument val4,
70                           QGenericArgument val5, QGenericArgument val6,
71                           QGenericArgument val7, QGenericArgument val8) {
72   // Now we try to find a handler for this message. BTW, I do love the Trolltech guys ;-)
73   // and now we even have a fast lookup! Thanks thiago!
74
75   QString handler = member.toLower();
76   handler[0] = handler[0].toUpper();
77
78   if(!handlerHash().contains(handler)) {
79     if(defaultHandler == -1) {
80       qWarning() << QString("No such Handler: %1::handle%2").arg(metaObject()->className(), handler);
81       return;
82     } else {
83       void *param[] = {0, Q_ARG(QString, member).data(), val0.data(), val1.data(), val2.data(), val3.data(), val4.data(),
84                        val5.data(), val6.data(), val7.data(), val8.data(), val8.data()};
85       qt_metacall(QMetaObject::InvokeMetaMethod, defaultHandler, param);
86       return;
87     }
88   }
89
90   void *param[] = {0, val0.data(), val1.data(), val2.data(), val3.data(), val4.data(),
91                    val5.data(), val6.data(), val7.data(), val8.data(), val8.data(), 0};
92   qt_metacall(QMetaObject::InvokeMetaMethod, handlerHash()[handler], param);
93 }
94
95 QString BasicHandler::serverDecode(const QByteArray &string) {
96   return networkConnection()->serverDecode(string);
97 }
98
99 QStringList BasicHandler::serverDecode(const QList<QByteArray> &stringlist) {
100   QStringList list;
101   foreach(QByteArray s, stringlist) list << networkConnection()->serverDecode(s);
102   return list;
103 }
104
105 QString BasicHandler::channelDecode(const QString &bufferName, const QByteArray &string) {
106   return networkConnection()->channelDecode(bufferName, string);
107 }
108
109 QStringList BasicHandler::channelDecode(const QString &bufferName, const QList<QByteArray> &stringlist) {
110   QStringList list;
111   foreach(QByteArray s, stringlist) list << networkConnection()->channelDecode(bufferName, s);
112   return list;
113 }
114
115 QString BasicHandler::userDecode(const QString &userNick, const QByteArray &string) {
116   return networkConnection()->userDecode(userNick, string);
117 }
118
119 QStringList BasicHandler::userDecode(const QString &userNick, const QList<QByteArray> &stringlist) {
120   QStringList list;
121   foreach(QByteArray s, stringlist) list << networkConnection()->userDecode(userNick, s);
122   return list;
123 }
124
125 /*** ***/
126
127 QByteArray BasicHandler::serverEncode(const QString &string) {
128   return networkConnection()->serverEncode(string);
129 }
130
131 QList<QByteArray> BasicHandler::serverEncode(const QStringList &stringlist) {
132   QList<QByteArray> list;
133   foreach(QString s, stringlist) list << networkConnection()->serverEncode(s);
134   return list;
135 }
136
137 QByteArray BasicHandler::channelEncode(const QString &bufferName, const QString &string) {
138   return networkConnection()->channelEncode(bufferName, string);
139 }
140
141 QList<QByteArray> BasicHandler::channelEncode(const QString &bufferName, const QStringList &stringlist) {
142   QList<QByteArray> list;
143   foreach(QString s, stringlist) list << networkConnection()->channelEncode(bufferName, s);
144   return list;
145 }
146
147 QByteArray BasicHandler::userEncode(const QString &userNick, const QString &string) {
148   return networkConnection()->userEncode(userNick, string);
149 }
150
151 QList<QByteArray> BasicHandler::userEncode(const QString &userNick, const QStringList &stringlist) {
152   QList<QByteArray> list;
153   foreach(QString s, stringlist) list << networkConnection()->userEncode(userNick, s);
154   return list;
155 }
156
157 // ====================
158 //  protected:
159 // ====================
160 BufferInfo::Type BasicHandler::typeByTarget(const QString &target) const {
161   if(target.isEmpty())
162     return BufferInfo::StatusBuffer;
163
164   if(network()->isChannelName(target))
165     return BufferInfo::ChannelBuffer;
166
167   return BufferInfo::QueryBuffer;
168 }
169
170 void BasicHandler::putCmd(const QString &cmd, const QByteArray &param, const QByteArray &prefix) {
171   QList<QByteArray> list;
172   list << param;
173   emit putCmd(cmd, list, prefix);
174 }
175
176 void BasicHandler::displayMsg(Message::Type msgType, QString target, QString text, QString sender, Message::Flags flags) {
177   IrcChannel *channel = network()->ircChannel(target);
178   if(!channel && (target.startsWith('$') || target.startsWith('#')))
179     target = nickFromMask(sender);
180
181   emit displayMsg(msgType, typeByTarget(target), target, text, sender, flags);
182 }