X-Git-Url: https://git.quassel-irc.org/?p=quassel.git;a=blobdiff_plain;f=src%2Fclient%2Fnetworkmodel.cpp;h=c25fda61c18c2d2c3afc294ab50ce7e679088955;hp=6e6397247322d3b3fdf1adc25a331f6c1dfbf1f2;hb=76ad6e4e368f8444adc4252f5abe9adfb1aed273;hpb=f824db0e31b54969e0b7fa0b5405b1e9173d482c diff --git a/src/client/networkmodel.cpp b/src/client/networkmodel.cpp index 6e639724..c25fda61 100644 --- a/src/client/networkmodel.cpp +++ b/src/client/networkmodel.cpp @@ -21,29 +21,31 @@ #include "networkmodel.h" #include +#include // for Qt::escape() #include "buffermodel.h" +#include "buffersettings.h" #include "client.h" -#include "signalproxy.h" -#include "network.h" +#include "clientignorelistmanager.h" +#include "clientsettings.h" #include "ircchannel.h" - -#include "buffersettings.h" - -#include "util.h" // get rid of this (needed for isChannelName) +#include "network.h" +#include "signalproxy.h" /***************************************** * Network Items *****************************************/ NetworkItem::NetworkItem(const NetworkId &netid, AbstractTreeItem *parent) : PropertyMapItem(QList() << "networkName" << "currentServer" << "nickCount", parent), - _networkId(netid) + _networkId(netid), + _statusBufferItem(0) { // DO NOT EMIT dataChanged() DIRECTLY IN NetworkItem // use networkDataChanged() instead. Otherwise you will end up in a infinite loop // as we "sync" the dataChanged() signals of NetworkItem and StatusBufferItem setFlags(Qt::ItemIsEnabled | Qt::ItemIsSelectable); connect(this, SIGNAL(networkDataChanged(int)), this, SIGNAL(dataChanged(int))); + connect(this, SIGNAL(beginRemoveChilds(int, int)), this, SLOT(onBeginRemoveChilds(int, int))); } QVariant NetworkItem::data(int column, int role) const { @@ -52,8 +54,8 @@ QVariant NetworkItem::data(int column, int role) const { case NetworkModel::BufferInfoRole: case NetworkModel::BufferTypeRole: case NetworkModel::BufferActivityRole: - if(childCount()) - return child(0)->data(column, role); + if(_statusBufferItem) + return _statusBufferItem->data(column, role); else return QVariant(); case NetworkModel::NetworkIdRole: @@ -67,6 +69,7 @@ QVariant NetworkItem::data(int column, int role) const { } } +// FIXME shouldn't we check the bufferItemCache here? BufferItem *NetworkItem::findBufferItem(BufferId bufferId) { BufferItem *bufferItem = 0; @@ -88,7 +91,8 @@ BufferItem *NetworkItem::bufferItem(const BufferInfo &bufferInfo) { switch(bufferInfo.type()) { case BufferInfo::StatusBuffer: - bufferItem = new StatusBufferItem(bufferInfo, this); + _statusBufferItem = new StatusBufferItem(bufferInfo, this); + bufferItem = _statusBufferItem; disconnect(this, SIGNAL(networkDataChanged(int)), this, SIGNAL(dataChanged(int))); connect(this, SIGNAL(networkDataChanged(int)), bufferItem, SIGNAL(dataChanged(int))); connect(bufferItem, SIGNAL(dataChanged(int)), this, SIGNAL(dataChanged(int))); @@ -104,6 +108,23 @@ BufferItem *NetworkItem::bufferItem(const BufferInfo &bufferInfo) { } newChild(bufferItem); + + // postprocess... this is necessary because Qt doesn't seem to like adding children which already have children on their own + switch(bufferInfo.type()) { + case BufferInfo::ChannelBuffer: + { + ChannelBufferItem *channelBufferItem = static_cast(bufferItem); + if(_network) { + IrcChannel *ircChannel = _network->ircChannel(bufferInfo.bufferName()); + if(ircChannel) + channelBufferItem->attachIrcChannel(ircChannel); + } + } + break; + default: + break; + } + return bufferItem; } @@ -114,17 +135,17 @@ void NetworkItem::attachNetwork(Network *network) { _network = network; connect(network, SIGNAL(networkNameSet(QString)), - this, SLOT(setNetworkName(QString))); + this, SLOT(setNetworkName(QString))); connect(network, SIGNAL(currentServerSet(QString)), - this, SLOT(setCurrentServer(QString))); + this, SLOT(setCurrentServer(QString))); connect(network, SIGNAL(ircChannelAdded(IrcChannel *)), - this, SLOT(attachIrcChannel(IrcChannel *))); + this, SLOT(attachIrcChannel(IrcChannel *))); connect(network, SIGNAL(ircUserAdded(IrcUser *)), - this, SLOT(attachIrcUser(IrcUser *))); + this, SLOT(attachIrcUser(IrcUser *))); connect(network, SIGNAL(connectedSet(bool)), - this, SIGNAL(networkDataChanged())); + this, SIGNAL(networkDataChanged())); connect(network, SIGNAL(destroyed()), - this, SIGNAL(networkDataChanged())); + this, SIGNAL(networkDataChanged())); emit networkDataChanged(); } @@ -171,8 +192,8 @@ void NetworkItem::setCurrentServer(const QString &serverName) { QString NetworkItem::toolTip(int column) const { Q_UNUSED(column); - QStringList toolTip(QString("%1").arg(networkName())); - toolTip.append(tr("Server: %1").arg(currentServer())); + QStringList toolTip(QString("%1").arg(Qt::escape(networkName()))); + toolTip.append(tr("Server: %1").arg(Qt::escape(currentServer()))); toolTip.append(tr("Users: %1").arg(nickCount())); if(_network) { @@ -182,6 +203,16 @@ QString NetworkItem::toolTip(int column) const { return QString("

%1

").arg(toolTip.join("
")); } +void NetworkItem::onBeginRemoveChilds(int start, int end) { + for(int i = start; i <= end; i++) { + StatusBufferItem *statusBufferItem = qobject_cast(child(i)); + if(statusBufferItem) { + _statusBufferItem = 0; + break; + } + } +} + /***************************************** * Fancy Buffer Items *****************************************/ @@ -202,7 +233,13 @@ void BufferItem::setActivityLevel(BufferInfo::ActivityLevel level) { void BufferItem::clearActivityLevel() { _activity = BufferInfo::NoActivity; - _lastSeenMarkerMsgId = _lastSeenMsgId; + _firstUnreadMsgId = MsgId(); + + // FIXME remove with core proto v11 + if(!(Client::coreFeatures() & Quassel::SynchronizedMarkerLine)) { + _markerLineMsgId = _lastSeenMsgId; + } + emit dataChanged(); } @@ -214,9 +251,19 @@ void BufferItem::updateActivityLevel(const Message &msg) { if(msg.flags() & Message::Self) // don't update activity for our own messages return; - if(lastSeenMsgId() >= msg.msgId()) + if (Client::ignoreListManager() + && Client::ignoreListManager()->match(msg, qobject_cast(parent())->networkName())) + return; + + if(msg.msgId() <= lastSeenMsgId()) return; + bool stateChanged = false; + if(!firstUnreadMsgId().isValid() || msg.msgId() < firstUnreadMsgId()) { + stateChanged = true; + _firstUnreadMsgId = msg.msgId(); + } + BufferInfo::ActivityLevel oldLevel = activityLevel(); _activity |= BufferInfo::OtherActivity; @@ -226,7 +273,9 @@ void BufferItem::updateActivityLevel(const Message &msg) { if(msg.flags() & Message::Highlight) _activity |= BufferInfo::Highlight; - if(oldLevel != _activity) + stateChanged |= (oldLevel != _activity); + + if(stateChanged) emit dataChanged(); } @@ -246,6 +295,10 @@ QVariant BufferItem::data(int column, int role) const { return isActive(); case NetworkModel::BufferActivityRole: return (int)activityLevel(); + case NetworkModel::BufferFirstUnreadMsgIdRole: + return qVariantFromValue(firstUnreadMsgId()); + case NetworkModel::MarkerLineMsgIdRole: + return qVariantFromValue(markerLineMsgId()); default: return PropertyMapItem::data(column, role); } @@ -267,14 +320,23 @@ void BufferItem::setBufferName(const QString &name) { emit dataChanged(0); } -void BufferItem::setLastSeenMsgId(const MsgId &msgId) { +void BufferItem::setLastSeenMsgId(MsgId msgId) { _lastSeenMsgId = msgId; - if(!isCurrentBuffer()) { - _lastSeenMarkerMsgId = msgId; + + // FIXME remove with core protocol v11 + if(!(Client::coreFeatures() & Quassel::SynchronizedMarkerLine)) { + if(!isCurrentBuffer()) + _markerLineMsgId = msgId; } + setActivityLevel(BufferInfo::NoActivity); } +void BufferItem::setMarkerLineMsgId(MsgId msgId) { + _markerLineMsgId = msgId; + emit dataChanged(); +} + bool BufferItem::isCurrentBuffer() const { return _bufferInfo.bufferId() == Client::bufferModel()->currentIndex().data(NetworkModel::BufferIdRole).value(); } @@ -293,13 +355,11 @@ StatusBufferItem::StatusBufferItem(const BufferInfo &bufferInfo, NetworkItem *pa } QString StatusBufferItem::toolTip(int column) const { - Q_UNUSED(column); - QStringList toolTip; - - QString netName = Client::network(bufferInfo().networkId())->networkName(); - toolTip.append(tr("Status buffer of %1").arg(netName)); - - return tr("

%1

").arg(toolTip.join("
")); + NetworkItem *networkItem = qobject_cast(parent()); + if(networkItem) + return networkItem->toolTip(column); + else + return QString(); } /***************************************** @@ -321,6 +381,8 @@ QueryBufferItem::QueryBufferItem(const BufferInfo &bufferInfo, NetworkItem *pare QVariant QueryBufferItem::data(int column, int role) const { switch(role) { + case Qt::EditRole: + return BufferItem::data(column, Qt::DisplayRole); case NetworkModel::IrcUserRole: return QVariant::fromValue(_ircUser); case NetworkModel::UserAwayRole: @@ -339,10 +401,10 @@ bool QueryBufferItem::setData(int column, const QVariant &value, int role) { { QString newName = value.toString(); if(!newName.isEmpty()) { - Client::renameBuffer(bufferId(), newName); - return true; + Client::renameBuffer(bufferId(), newName); + return true; } else { - return false; + return false; } } break; @@ -367,7 +429,7 @@ QString QueryBufferItem::toolTip(int column) const { toolTip.append(tr("Query with %1").arg(bufferName())); if(_ircUser) { - if(_ircUser->userModes() != "") toolTip[0].append(QString(" (%1)").arg(_ircUser->userModes())); + if(_ircUser->userModes() != "") toolTip[0].append(QString(" (+%1)").arg(_ircUser->userModes())); if(_ircUser->isAway()) { toolTip[0].append(QString(" (away%1)").arg(!_ircUser->awayMessage().isEmpty() ? (QString(" ") + _ircUser->awayMessage()) : QString())); } @@ -423,13 +485,6 @@ ChannelBufferItem::ChannelBufferItem(const BufferInfo &bufferInfo, AbstractTreeI : BufferItem(bufferInfo, parent), _ircChannel(0) { - const Network *net = Client::network(bufferInfo.networkId()); - if(!net) - return; - - IrcChannel *ircChannel = net->ircChannel(bufferInfo.bufferName()); - if(ircChannel) - attachIrcChannel(ircChannel); } QVariant ChannelBufferItem::data(int column, int role) const { @@ -445,7 +500,7 @@ QString ChannelBufferItem::toolTip(int column) const { Q_UNUSED(column); QStringList toolTip; - toolTip.append(tr("Channel %1").arg(bufferName())); + toolTip.append(tr("Channel %1").arg(Qt::escape(bufferName()))); if(isActive()) { //TODO: add channel modes toolTip.append(tr("Users: %1").arg(nickCount())); @@ -455,14 +510,13 @@ QString ChannelBufferItem::toolTip(int column) const { toolTip.append(tr("Mode: %1").arg(channelMode)); } - BufferSettings s; - bool showTopic = s.value("DisplayTopicInTooltip", QVariant(false)).toBool(); + ItemViewSettings s; + bool showTopic = s.displayTopicInTooltip(); if(showTopic) { QString _topic = topic(); if(_topic != "") { _topic = stripFormatCodes(_topic); - _topic.replace(QString("<"), QString("<")); - _topic.replace(QString(">"), QString(">")); + _topic = Qt::escape(_topic); toolTip.append(QString(" ")); toolTip.append(tr("Topic: %1").arg(_topic)); } @@ -480,19 +534,19 @@ void ChannelBufferItem::attachIrcChannel(IrcChannel *ircChannel) { _ircChannel = ircChannel; connect(ircChannel, SIGNAL(topicSet(QString)), - this, SLOT(setTopic(QString))); + this, SLOT(setTopic(QString))); connect(ircChannel, SIGNAL(ircUsersJoined(QList)), - this, SLOT(join(QList))); + this, SLOT(join(QList))); connect(ircChannel, SIGNAL(ircUserParted(IrcUser *)), - this, SLOT(part(IrcUser *))); + this, SLOT(part(IrcUser *))); connect(ircChannel, SIGNAL(parted()), - this, SLOT(ircChannelParted())); + this, SLOT(ircChannelParted())); connect(ircChannel, SIGNAL(ircUserModesSet(IrcUser *, QString)), - this, SLOT(userModeChanged(IrcUser *))); + this, SLOT(userModeChanged(IrcUser *))); connect(ircChannel, SIGNAL(ircUserModeAdded(IrcUser *, QString)), - this, SLOT(userModeChanged(IrcUser *))); + this, SLOT(userModeChanged(IrcUser *))); connect(ircChannel, SIGNAL(ircUserModeRemoved(IrcUser *, QString)), - this, SLOT(userModeChanged(IrcUser *))); + this, SLOT(userModeChanged(IrcUser *))); if(!ircChannel->ircUsers().isEmpty()) join(ircChannel->ircUsers()); @@ -580,7 +634,7 @@ void ChannelBufferItem::removeUserFromCategory(IrcUser *ircUser) { categoryItem = qobject_cast(child(i)); if(categoryItem->removeUser(ircUser)) { if(categoryItem->childCount() == 0) - removeChild(i); + removeChild(i); break; } } @@ -749,7 +803,7 @@ QString IrcUserItem::toolTip(int column) const { QStringList toolTip(QString("%1").arg(nickName())); if(_ircUser->userModes() != "") toolTip[0].append(QString(" (%1)").arg(_ircUser->userModes())); if(_ircUser->isAway()) { - toolTip[0].append(" is away"); + toolTip[0].append(tr(" is away")); if(!_ircUser->awayMessage().isEmpty()) toolTip[0].append(QString(" (%1)").arg(_ircUser->awayMessage())); } @@ -782,14 +836,20 @@ NetworkModel::NetworkModel(QObject *parent) : TreeModel(NetworkModel::defaultHeader(), parent) { connect(this, SIGNAL(rowsInserted(const QModelIndex &, int, int)), - this, SLOT(checkForNewBuffers(const QModelIndex &, int, int))); + this, SLOT(checkForNewBuffers(const QModelIndex &, int, int))); connect(this, SIGNAL(rowsAboutToBeRemoved(const QModelIndex &, int, int)), - this, SLOT(checkForRemovedBuffers(const QModelIndex &, int, int))); + this, SLOT(checkForRemovedBuffers(const QModelIndex &, int, int))); + + BufferSettings defaultSettings; + defaultSettings.notify("UserNoticesTarget", this, SLOT(messageRedirectionSettingsChanged())); + defaultSettings.notify("ServerNoticesTarget", this, SLOT(messageRedirectionSettingsChanged())); + defaultSettings.notify("ErrorMsgsTarget", this, SLOT(messageRedirectionSettingsChanged())); + messageRedirectionSettingsChanged(); } QListNetworkModel::defaultHeader() { QList data; - data << tr("Buffer") << tr("Topic") << tr("Nick Count"); + data << tr("Chat") << tr("Topic") << tr("Nick Count"); return data; } @@ -941,27 +1001,100 @@ MsgId NetworkModel::lastSeenMsgId(BufferId bufferId) const { return _bufferItemCache[bufferId]->lastSeenMsgId(); } -MsgId NetworkModel::lastSeenMarkerMsgId(BufferId bufferId) const { +MsgId NetworkModel::markerLineMsgId(BufferId bufferId) const { if(!_bufferItemCache.contains(bufferId)) return MsgId(); - return _bufferItemCache[bufferId]->lastSeenMarkerMsgId(); + return _bufferItemCache[bufferId]->markerLineMsgId(); +} + +// FIXME we always seem to use this (expensive) non-const version +MsgId NetworkModel::lastSeenMsgId(const BufferId &bufferId) { + BufferItem *bufferItem = findBufferItem(bufferId); + if(!bufferItem) { + qDebug() << "NetworkModel::lastSeenMsgId(): buffer is unknown:" << bufferId; + Client::purgeKnownBufferIds(); + return MsgId(); + } + return bufferItem->lastSeenMsgId(); } void NetworkModel::setLastSeenMsgId(const BufferId &bufferId, const MsgId &msgId) { BufferItem *bufferItem = findBufferItem(bufferId); if(!bufferItem) { qDebug() << "NetworkModel::setLastSeenMsgId(): buffer is unknown:" << bufferId; + Client::purgeKnownBufferIds(); return; } bufferItem->setLastSeenMsgId(msgId); + emit lastSeenMsgSet(bufferId, msgId); } -void NetworkModel::updateBufferActivity(const Message &msg) { - BufferItem *item = bufferItem(msg.bufferInfo()); - item->updateActivityLevel(msg); - if(item->isCurrentBuffer()) - emit setLastSeenMsg(item->bufferId(), msg.msgId()); +void NetworkModel::setMarkerLineMsgId(const BufferId &bufferId, const MsgId &msgId) { + BufferItem *bufferItem = findBufferItem(bufferId); + if(!bufferItem) { + qDebug() << "NetworkModel::setMarkerLineMsgId(): buffer is unknown:" << bufferId; + Client::purgeKnownBufferIds(); + return; + } + bufferItem->setMarkerLineMsgId(msgId); + emit markerLineSet(bufferId, msgId); +} + +void NetworkModel::updateBufferActivity(Message &msg) { + int redirectionTarget = 0; + switch(msg.type()) { + case Message::Notice: + if(bufferType(msg.bufferId()) != BufferInfo::ChannelBuffer) { + msg.setFlags(msg.flags() | Message::Redirected); + if(msg.flags() & Message::ServerMsg) { + // server notice + redirectionTarget = _serverNoticesTarget; + } else { + redirectionTarget = _userNoticesTarget; + } + } + break; + case Message::Error: + msg.setFlags(msg.flags() | Message::Redirected); + redirectionTarget = _errorMsgsTarget; + break; + // Update IrcUser's last activity + case Message::Plain: + case Message::Action: + if(bufferType(msg.bufferId()) == BufferInfo::ChannelBuffer) { + const Network *net = Client::network(msg.bufferInfo().networkId()); + IrcUser *user = net ? net->ircUser(nickFromMask(msg.sender())) : 0; + if(user) + user->setLastChannelActivity(msg.bufferId(), msg.timestamp()); + } + break; + default: + break; + } + + if(msg.flags() & Message::Redirected) { + if(redirectionTarget & BufferSettings::DefaultBuffer) + updateBufferActivity(bufferItem(msg.bufferInfo()), msg); + + if(redirectionTarget & BufferSettings::StatusBuffer) { + const NetworkItem *netItem = findNetworkItem(msg.bufferInfo().networkId()); + if(netItem) { + updateBufferActivity(netItem->statusBufferItem(), msg); + } + } + } else { + updateBufferActivity(bufferItem(msg.bufferInfo()), msg); + } +} + +void NetworkModel::updateBufferActivity(BufferItem *bufferItem, const Message &msg) { + if(!bufferItem) + return; + + bufferItem->updateActivityLevel(msg); + if(bufferItem->isCurrentBuffer()) + emit requestSetLastSeenMsg(bufferItem->bufferId(), msg.msgId()); } void NetworkModel::setBufferActivity(const BufferId &bufferId, BufferInfo::ActivityLevel level) { @@ -1097,3 +1230,10 @@ bool NetworkModel::bufferItemLessThan(const BufferItem *left, const BufferItem * return QString::compare(left->bufferName(), right->bufferName(), Qt::CaseInsensitive) < 0; } +void NetworkModel::messageRedirectionSettingsChanged() { + BufferSettings bufferSettings; + + _userNoticesTarget = bufferSettings.userNoticesTarget(); + _serverNoticesTarget = bufferSettings.serverNoticesTarget(); + _errorMsgsTarget = bufferSettings.errorMsgsTarget(); +}