Made the Nicklist Pretty again
authorMarcus Eggenberger <egs@quassel-irc.org>
Thu, 17 Jan 2008 13:38:37 +0000 (13:38 +0000)
committerMarcus Eggenberger <egs@quassel-irc.org>
Thu, 17 Jan 2008 13:38:37 +0000 (13:38 +0000)
src/client/networkmodel.cpp
src/client/networkmodel.h

index ff44942..61ee7b9 100644 (file)
@@ -120,6 +120,12 @@ void BufferItem::attachIrcChannel(IrcChannel *ircChannel) {
          this, SLOT(part(IrcUser *)));
   connect(ircChannel, SIGNAL(destroyed()),
          this, SLOT(ircChannelDestroyed()));
+  connect(ircChannel, SIGNAL(ircUserModesSet(IrcUser *, QString)),
+         this, SLOT(userModeChanged(IrcUser *)));
+  connect(ircChannel, SIGNAL(ircUserModeAdded(IrcUser *, QString)),
+         this, SLOT(userModeChanged(IrcUser *)));
+  connect(ircChannel, SIGNAL(ircUserModeRemoved(IrcUser *, QString)),
+         this, SLOT(userModeChanged(IrcUser *)));
   
   emit dataChanged();
 }
@@ -159,15 +165,48 @@ void BufferItem::setTopic(const QString &topic) {
 }
 
 void BufferItem::join(IrcUser *ircUser) {
-  emit newChild(new IrcUserItem(ircUser, this));
+  if(!ircUser)
+    return;
+
+  addUserToCategory(ircUser);
   emit dataChanged(2);
 }
 
+void BufferItem::addUserToCategory(IrcUser *ircUser) {
+  Q_ASSERT(_ircChannel);
+  
+  UserCategoryItem *categoryItem;
+  int categoryId = UserCategoryItem::categoryFromModes(_ircChannel->userModes(ircUser));
+  if(!(categoryItem = qobject_cast<UserCategoryItem *>(childById(qHash(categoryId))))) {
+    categoryItem = new UserCategoryItem(categoryId, this);
+    emit newChild(categoryItem);
+  }
+  
+  categoryItem->addUser(ircUser);
+}
+
 void BufferItem::part(IrcUser *ircUser) {
   Q_UNUSED(ircUser);
   emit dataChanged(2);
 }
 
+void BufferItem::removeUserFromCategory(IrcUser *ircUser) {
+  UserCategoryItem *categoryItem = 0;
+  IrcUserItem *userItem;
+  for(int i = 0; i < childCount(); i++) {
+    categoryItem = qobject_cast<UserCategoryItem *>(child(i));
+    if(userItem = qobject_cast<IrcUserItem *>(categoryItem->childById((quint64)ircUser))) {
+      userItem->deleteLater();
+      return;
+    }
+  }
+}
+
+void BufferItem::userModeChanged(IrcUser *ircUser) {
+  removeUserFromCategory(ircUser);
+  addUserToCategory(ircUser);
+}
+
 /*****************************************
 *  Network Items
 *****************************************/
@@ -265,6 +304,54 @@ void NetworkItem::setCurrentServer(const QString &serverName) {
   emit dataChanged(1);
 }
 
+/*****************************************
+*  User Category Items (like @vh etc.)
+*****************************************/
+// we hardcode this even though we have PREFIX in network... but that wouldn't help with mapping modes to
+// category strings anyway.
+const QList<UserCategoryItem::Category> UserCategoryItem::categories = QList<UserCategoryItem::Category>() << UserCategoryItem::Category('q', "Owners")
+                                                                                                          << UserCategoryItem::Category('a', "Admins")
+                                                                                                          << UserCategoryItem::Category('a', "Admins")
+                                                                                                          << UserCategoryItem::Category('o', "Operators")
+                                                                                                          << UserCategoryItem::Category('h', "Half-Ops")
+                                                                                                          << UserCategoryItem::Category('v', "Voiced");
+
+UserCategoryItem::UserCategoryItem(int category, AbstractTreeItem *parent)
+  : PropertyMapItem(QStringList() << "categoryId", parent),
+    _category(category)
+{
+  connect(this, SIGNAL(childDestroyed(int)),
+         this, SLOT(checkNoChilds()));
+}
+
+QString UserCategoryItem::categoryId() {
+  if(_category < categories.count())
+    return categories[_category].displayString;
+  else
+    return QString("Users");
+}
+
+void UserCategoryItem::checkNoChilds() {
+  if(childCount() == 0)
+    deleteLater();
+}
+
+quint64 UserCategoryItem::id() const {
+  return qHash(_category);
+}
+
+void UserCategoryItem::addUser(IrcUser *ircUser) {
+  emit newChild(new IrcUserItem(ircUser, this));
+}
+
+int UserCategoryItem::categoryFromModes(const QString &modes) {
+  for(int i = 0; i < categories.count(); i++) {
+    if(modes.contains(categories[i].mode))
+      return i;
+  }
+  return categories.count();
+}
+     
 /*****************************************
 *  Irc User Items
 *****************************************/
@@ -283,6 +370,14 @@ QString IrcUserItem::nickName() {
   return _ircUser->nick();
 }
 
+IrcUser *IrcUserItem::ircUser() {
+  return _ircUser;
+}
+
+quint64 IrcUserItem::id() const {
+  return (quint64)_ircUser;
+}
+
 void IrcUserItem::setNick(QString newNick) {
   Q_UNUSED(newNick);
   emit dataChanged(0);
@@ -291,7 +386,6 @@ void IrcUserItem::ircUserDestroyed() {
   deleteLater();
 }
 
-
 /*****************************************
  * NetworkModel
  *****************************************/
index 09e1bd2..a9a108d 100644 (file)
@@ -88,6 +88,10 @@ public slots:
   void join(IrcUser *ircUser);
   void part(IrcUser *ircUser);
 
+  void addUserToCategory(IrcUser *ircUser);
+  void removeUserFromCategory(IrcUser *ircUser);
+  void userModeChanged(IrcUser *ircUser);
+                                        
 private slots:
   void ircChannelDestroyed();
   
@@ -135,6 +139,38 @@ private:
   QPointer<Network> _network;
 };
 
+/*****************************************
+*  User Category Items (like @vh etc.)
+*****************************************/
+class UserCategoryItem : public PropertyMapItem {
+  Q_OBJECT
+  Q_PROPERTY(QString categoryId READ categoryId)
+    
+public:
+  UserCategoryItem(int category, AbstractTreeItem *parent);
+
+  QString categoryId();
+  virtual quint64 id() const;
+
+  void addUser(IrcUser *ircUser);
+
+  static int categoryFromModes(const QString &modes);
+
+private slots:
+  void checkNoChilds();
+
+private:
+  int _category;
+
+  struct Category {
+    QChar mode;
+    QString displayString;
+    inline Category(QChar mode_, QString displayString_) : mode(mode_), displayString(displayString_) {};
+  };
+
+  static const QList<Category> categories;
+};
+
 /*****************************************
 *  Irc User Items
 *****************************************/
@@ -146,6 +182,8 @@ public:
   IrcUserItem(IrcUser *ircUser, AbstractTreeItem *parent);
 
   QString nickName();
+  IrcUser *ircUser();
+  virtual quint64 id() const;
 
 private slots:
   void setNick(QString newNick);