/***************************************************************************
- * Copyright (C) 2005-09 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 "ircuser.h"
_server(),
// _idleTime(QDateTime::currentDateTime()),
_ircOperator(),
- _lastAwayMessage(0),
+ _lastAwayMessageTime(),
_whoisServiceReply(),
+ _encrypted(false),
_network(network),
_codecForEncoding(0),
_codecForDecoding(0)
{
- updateObjectName();
+ updateObjectName();
+ _lastAwayMessageTime.setTimeSpec(Qt::UTC);
+ _lastAwayMessageTime.setMSecsSinceEpoch(0);
}
-IrcUser::~IrcUser() {
+
+IrcUser::~IrcUser()
+{
}
+
// ====================
// PUBLIC:
// ====================
-QString IrcUser::hostmask() const {
- return QString("%1!%2@%3").arg(nick()).arg(user()).arg(host());
+QString IrcUser::hostmask() const
+{
+ return QString("%1!%2@%3").arg(nick()).arg(user()).arg(host());
}
-QDateTime IrcUser::idleTime() {
- if(QDateTime::currentDateTime().toTime_t() - _idleTimeSet.toTime_t() > 1200)
- _idleTime = QDateTime();
- return _idleTime;
+
+QDateTime IrcUser::idleTime()
+{
+ if ((QDateTime::currentDateTime().toMSecsSinceEpoch() - _idleTimeSet.toMSecsSinceEpoch())
+ > 1200000) {
+ // 20 * 60 * 1000 = 1200000
+ // 20 minutes have elapsed, clear the known idle time as it's likely inaccurate by now
+ _idleTime = QDateTime();
+ }
+ return _idleTime;
}
-QStringList IrcUser::channels() const {
- QStringList chanList;
- IrcChannel *channel;
- foreach(channel, _channels) {
- chanList << channel->name();
- }
- return chanList;
+
+QStringList IrcUser::channels() const
+{
+ QStringList chanList;
+ IrcChannel *channel;
+ foreach(channel, _channels) {
+ chanList << channel->name();
+ }
+ return chanList;
}
-void IrcUser::setCodecForEncoding(const QString &name) {
- setCodecForEncoding(QTextCodec::codecForName(name.toAscii()));
+void IrcUser::setCodecForEncoding(const QString &name)
+{
+ setCodecForEncoding(QTextCodec::codecForName(name.toLatin1()));
}
-void IrcUser::setCodecForEncoding(QTextCodec *codec) {
- _codecForEncoding = codec;
+
+void IrcUser::setCodecForEncoding(QTextCodec *codec)
+{
+ _codecForEncoding = codec;
}
-void IrcUser::setCodecForDecoding(const QString &name) {
- setCodecForDecoding(QTextCodec::codecForName(name.toAscii()));
+
+void IrcUser::setCodecForDecoding(const QString &name)
+{
+ setCodecForDecoding(QTextCodec::codecForName(name.toLatin1()));
}
-void IrcUser::setCodecForDecoding(QTextCodec *codec) {
- _codecForDecoding = codec;
+
+void IrcUser::setCodecForDecoding(QTextCodec *codec)
+{
+ _codecForDecoding = codec;
}
-QString IrcUser::decodeString(const QByteArray &text) const {
- if(!codecForDecoding()) return network()->decodeString(text);
- return ::decodeString(text, codecForDecoding());
+
+QString IrcUser::decodeString(const QByteArray &text) const
+{
+ if (!codecForDecoding()) return network()->decodeString(text);
+ return ::decodeString(text, codecForDecoding());
}
-QByteArray IrcUser::encodeString(const QString &string) const {
- if(codecForEncoding()) {
- return codecForEncoding()->fromUnicode(string);
- }
- return network()->encodeString(string);
+
+QByteArray IrcUser::encodeString(const QString &string) const
+{
+ if (codecForEncoding()) {
+ return codecForEncoding()->fromUnicode(string);
+ }
+ return network()->encodeString(string);
}
+
// ====================
// PUBLIC SLOTS:
// ====================
-void IrcUser::setUser(const QString &user) {
- if(!user.isEmpty() && _user != user) {
- _user = user;
- SYNC(ARG(user));
- }
+void IrcUser::setUser(const QString &user)
+{
+ if (!user.isEmpty() && _user != user) {
+ _user = user;
+ SYNC(ARG(user));
+ }
}
-void IrcUser::setRealName(const QString &realName) {
- if (!realName.isEmpty() && _realName != realName) {
- _realName = realName;
- SYNC(ARG(realName))
- }
+
+void IrcUser::setRealName(const QString &realName)
+{
+ if (!realName.isEmpty() && _realName != realName) {
+ _realName = realName;
+ SYNC(ARG(realName))
+ }
}
-void IrcUser::setAway(const bool &away) {
- if(away != _away) {
- _away = away;
- emit awaySet(away);
- SYNC(ARG(away))
- }
+
+void IrcUser::setAccount(const QString &account)
+{
+ if (_account != account) {
+ _account = account;
+ SYNC(ARG(account))
+ }
}
-void IrcUser::setAwayMessage(const QString &awayMessage) {
- if(!awayMessage.isEmpty() && _awayMessage != awayMessage) {
- _awayMessage = awayMessage;
- SYNC(ARG(awayMessage))
- }
+
+void IrcUser::setAway(const bool &away)
+{
+ if (away != _away) {
+ _away = away;
+ SYNC(ARG(away))
+ emit awaySet(away);
+ }
}
-void IrcUser::setIdleTime(const QDateTime &idleTime) {
- if(idleTime.isValid() && _idleTime != idleTime) {
- _idleTime = idleTime;
- _idleTimeSet = QDateTime::currentDateTime();
- SYNC(ARG(idleTime))
- }
+
+void IrcUser::setAwayMessage(const QString &awayMessage)
+{
+ if (!awayMessage.isEmpty() && _awayMessage != awayMessage) {
+ _awayMessage = awayMessage;
+ SYNC(ARG(awayMessage))
+ }
}
-void IrcUser::setLoginTime(const QDateTime &loginTime) {
- if(loginTime.isValid() && _loginTime != loginTime) {
- _loginTime = loginTime;
- SYNC(ARG(loginTime))
- }
+
+void IrcUser::setIdleTime(const QDateTime &idleTime)
+{
+ if (idleTime.isValid() && _idleTime != idleTime) {
+ _idleTime = idleTime;
+ _idleTimeSet = QDateTime::currentDateTime();
+ SYNC(ARG(idleTime))
+ }
}
-void IrcUser::setServer(const QString &server) {
- if(!server.isEmpty() && _server != server) {
- _server = server;
- SYNC(ARG(server))
- }
+
+void IrcUser::setLoginTime(const QDateTime &loginTime)
+{
+ if (loginTime.isValid() && _loginTime != loginTime) {
+ _loginTime = loginTime;
+ SYNC(ARG(loginTime))
+ }
}
-void IrcUser::setIrcOperator(const QString &ircOperator) {
- if(!ircOperator.isEmpty() && _ircOperator != ircOperator) {
- _ircOperator = ircOperator;
- SYNC(ARG(ircOperator))
- }
+
+void IrcUser::setServer(const QString &server)
+{
+ if (!server.isEmpty() && _server != server) {
+ _server = server;
+ SYNC(ARG(server))
+ }
}
-void IrcUser::setLastAwayMessage(const int &lastAwayMessage) {
- if(lastAwayMessage > _lastAwayMessage) {
- _lastAwayMessage = lastAwayMessage;
- SYNC(ARG(lastAwayMessage))
- }
+
+void IrcUser::setIrcOperator(const QString &ircOperator)
+{
+ if (!ircOperator.isEmpty() && _ircOperator != ircOperator) {
+ _ircOperator = ircOperator;
+ SYNC(ARG(ircOperator))
+ }
}
-void IrcUser::setHost(const QString &host) {
- if(!host.isEmpty() && _host != host) {
- _host = host;
- SYNC(ARG(host))
- }
+
+// This function is only ever called by SYNC calls from legacy cores (pre-0.13).
+// Therefore, no SYNC call is needed here.
+void IrcUser::setLastAwayMessage(const int &lastAwayMessage)
+{
+ QDateTime lastAwayMessageTime = QDateTime();
+ lastAwayMessageTime.setTimeSpec(Qt::UTC);
+#if QT_VERSION >= 0x050800
+ lastAwayMessageTime.fromSecsSinceEpoch(lastAwayMessage);
+#else
+ // toSecsSinceEpoch() was added in Qt 5.8. Manually downconvert to seconds for now.
+ // See https://doc.qt.io/qt-5/qdatetime.html#toMSecsSinceEpoch
+ lastAwayMessageTime.fromMSecsSinceEpoch(lastAwayMessage * 1000);
+#endif
+ setLastAwayMessageTime(lastAwayMessageTime);
}
-void IrcUser::setNick(const QString &nick) {
- if(!nick.isEmpty() && nick != _nick) {
- _nick = nick;
- updateObjectName();
- emit nickSet(nick);
- SYNC(ARG(nick))
- }
+
+void IrcUser::setLastAwayMessageTime(const QDateTime &lastAwayMessageTime)
+{
+ if (lastAwayMessageTime > _lastAwayMessageTime) {
+ _lastAwayMessageTime = lastAwayMessageTime;
+ SYNC(ARG(lastAwayMessageTime))
+ }
}
-void IrcUser::setWhoisServiceReply(const QString &whoisServiceReply) {
- if(!whoisServiceReply.isEmpty() && whoisServiceReply != _whoisServiceReply) {
- _whoisServiceReply = whoisServiceReply;
- SYNC(ARG(whoisServiceReply))
- }
+
+void IrcUser::setHost(const QString &host)
+{
+ if (!host.isEmpty() && _host != host) {
+ _host = host;
+ SYNC(ARG(host))
+ }
}
-void IrcUser::setSuserHost(const QString &suserHost) {
- if(!suserHost.isEmpty() && suserHost != _suserHost) {
- _suserHost = suserHost;
- SYNC(ARG(suserHost))
- }
+
+void IrcUser::setNick(const QString &nick)
+{
+ if (!nick.isEmpty() && nick != _nick) {
+ _nick = nick;
+ updateObjectName();
+ SYNC(ARG(nick))
+ emit nickSet(nick);
+ }
}
-void IrcUser::updateObjectName() {
- renameObject(QString::number(network()->networkId().toInt()) + "/" + _nick);
+
+void IrcUser::setWhoisServiceReply(const QString &whoisServiceReply)
+{
+ if (!whoisServiceReply.isEmpty() && whoisServiceReply != _whoisServiceReply) {
+ _whoisServiceReply = whoisServiceReply;
+ SYNC(ARG(whoisServiceReply))
+ }
}
-void IrcUser::updateHostmask(const QString &mask) {
- if(mask == hostmask())
- return;
- QString user = userFromMask(mask);
- QString host = hostFromMask(mask);
- setUser(user);
- setHost(host);
+void IrcUser::setSuserHost(const QString &suserHost)
+{
+ if (!suserHost.isEmpty() && suserHost != _suserHost) {
+ _suserHost = suserHost;
+ SYNC(ARG(suserHost))
+ }
+}
+
+
+void IrcUser::setEncrypted(bool encrypted)
+{
+ _encrypted = encrypted;
+ emit encryptedSet(encrypted);
+ SYNC(ARG(encrypted))
}
-void IrcUser::joinChannel(IrcChannel *channel) {
- Q_ASSERT(channel);
- if(!_channels.contains(channel)) {
- _channels.insert(channel);
- channel->joinIrcUser(this);
- }
+
+void IrcUser::updateObjectName()
+{
+ renameObject(QString::number(network()->networkId().toInt()) + "/" + _nick);
+}
+
+
+void IrcUser::updateHostmask(const QString &mask)
+{
+ if (mask == hostmask())
+ return;
+
+ QString user = userFromMask(mask);
+ QString host = hostFromMask(mask);
+ setUser(user);
+ setHost(host);
+}
+
+
+void IrcUser::joinChannel(IrcChannel *channel, bool skip_channel_join)
+{
+ Q_ASSERT(channel);
+ if (!_channels.contains(channel)) {
+ _channels.insert(channel);
+ if (!skip_channel_join)
+ channel->joinIrcUser(this);
+ }
}
-void IrcUser::joinChannel(const QString &channelname) {
- joinChannel(network()->newIrcChannel(channelname));
+
+void IrcUser::joinChannel(const QString &channelname)
+{
+ joinChannel(network()->newIrcChannel(channelname));
}
-void IrcUser::partChannel(IrcChannel *channel) {
- if(_channels.contains(channel)) {
- _channels.remove(channel);
- disconnect(channel, 0, this, 0);
- channel->part(this);
- QString channelName = channel->name();
- SYNC_OTHER(partChannel, ARG(channelName))
- if(_channels.isEmpty() && !network()->isMe(this))
- quit();
- }
+
+void IrcUser::partChannel(IrcChannel *channel)
+{
+ if (_channels.contains(channel)) {
+ _channels.remove(channel);
+ disconnect(channel, 0, this, 0);
+ channel->part(this);
+ QString channelName = channel->name();
+ SYNC_OTHER(partChannel, ARG(channelName))
+ if (_channels.isEmpty() && !network()->isMe(this))
+ quit();
+ }
}
-void IrcUser::partChannel(const QString &channelname) {
- IrcChannel *channel = network()->ircChannel(channelname);
- if(channel == 0) {
- qWarning() << "IrcUser::partChannel(): received part for unknown Channel" << channelname;
- } else {
- partChannel(channel);
- }
+
+void IrcUser::partChannel(const QString &channelname)
+{
+ IrcChannel *channel = network()->ircChannel(channelname);
+ if (channel == 0) {
+ qWarning() << "IrcUser::partChannel(): received part for unknown Channel" << channelname;
+ }
+ else {
+ partChannel(channel);
+ }
}
-void IrcUser::quit() {
- QList<IrcChannel *> channels = _channels.toList();
- _channels.clear();
- foreach(IrcChannel *channel, channels) {
- disconnect(channel, 0, this, 0);
- channel->part(this);
- }
- network()->removeIrcUser(this);
- emit quited();
+
+void IrcUser::quit()
+{
+ QList<IrcChannel *> channels = _channels.toList();
+ _channels.clear();
+ foreach(IrcChannel *channel, channels) {
+ disconnect(channel, 0, this, 0);
+ channel->part(this);
+ }
+ network()->removeIrcUser(this);
+ SYNC(NO_ARG)
+ emit quited();
}
-void IrcUser::channelDestroyed() {
- // private slot!
- IrcChannel *channel = static_cast<IrcChannel*>(sender());
- if(_channels.contains(channel)) {
- _channels.remove(channel);
- if(_channels.isEmpty() && !network()->isMe(this))
- quit();
- }
+
+void IrcUser::channelDestroyed()
+{
+ // private slot!
+ IrcChannel *channel = static_cast<IrcChannel *>(sender());
+ if (_channels.contains(channel)) {
+ _channels.remove(channel);
+ if (_channels.isEmpty() && !network()->isMe(this))
+ quit();
+ }
}
-void IrcUser::setUserModes(const QString &modes) {
- _userModes = modes;
- SYNC(ARG(modes))
- emit userModesSet(modes);
+
+void IrcUser::setUserModes(const QString &modes)
+{
+ if (_userModes != modes) {
+ _userModes = modes;
+ SYNC(ARG(modes))
+ emit userModesSet(modes);
+ }
}
-void IrcUser::addUserModes(const QString &modes) {
- if(modes.isEmpty())
- return;
- for(int i = 0; i < modes.count(); i++) {
- if(!_userModes.contains(modes[i]))
- _userModes += modes[i];
- }
+void IrcUser::addUserModes(const QString &modes)
+{
+ if (modes.isEmpty())
+ return;
- SYNC(ARG(modes))
- emit userModesAdded(modes);
+ // Don't needlessly sync when no changes are made
+ bool changesMade = false;
+ for (int i = 0; i < modes.count(); i++) {
+ if (!_userModes.contains(modes[i])) {
+ _userModes += modes[i];
+ changesMade = true;
+ }
+ }
+
+ if (changesMade) {
+ SYNC(ARG(modes))
+ emit userModesAdded(modes);
+ }
}
-void IrcUser::removeUserModes(const QString &modes) {
- if(modes.isEmpty())
- return;
- for(int i = 0; i < modes.count(); i++) {
- _userModes.remove(modes[i]);
- }
- SYNC(ARG(modes))
- emit userModesRemoved(modes);
+void IrcUser::removeUserModes(const QString &modes)
+{
+ if (modes.isEmpty())
+ return;
+
+ for (int i = 0; i < modes.count(); i++) {
+ _userModes.remove(modes[i]);
+ }
+ SYNC(ARG(modes))
+ emit userModesRemoved(modes);
}
-void IrcUser::setLastChannelActivity(BufferId buffer, const QDateTime &time) {
- _lastActivity[buffer] = time;
- emit lastChannelActivityUpdated(buffer, time);
+
+void IrcUser::setLastChannelActivity(BufferId buffer, const QDateTime &time)
+{
+ _lastActivity[buffer] = time;
+ emit lastChannelActivityUpdated(buffer, time);
}
-void IrcUser::setLastSpokenTo(BufferId buffer, const QDateTime &time) {
- _lastSpokenTo[buffer] = time;
- emit lastSpokenToUpdated(buffer, time);
+
+void IrcUser::setLastSpokenTo(BufferId buffer, const QDateTime &time)
+{
+ _lastSpokenTo[buffer] = time;
+ emit lastSpokenToUpdated(buffer, time);
}