/***************************************************************************
- * Copyright (C) 2005-08 by the Quassel Project *
+ * Copyright (C) 2005-2013 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 "nickview.h"
+
+#include <QApplication>
#include <QHeaderView>
+#include <QScrollBar>
#include <QDebug>
#include <QMenu>
+#include "buffermodel.h"
+#include "client.h"
+#include "contextmenuactionprovider.h"
+#include "graphicalui.h"
#include "nickview.h"
#include "nickviewfilter.h"
#include "networkmodel.h"
#include "types.h"
-#include "client.h"
-
NickView::NickView(QWidget *parent)
- : QTreeView(parent)
+ : QTreeView(parent)
{
- setIndentation(10);
- setAnimated(true);
- header()->hide();
- setSortingEnabled(true);
- sortByColumn(0, Qt::AscendingOrder);
-
- setContextMenuPolicy(Qt::CustomContextMenu);
-
- connect(this, SIGNAL(customContextMenuRequested(const QPoint&)),
- this, SLOT(showContextMenu(const QPoint&)));
- connect(this, SIGNAL(activated( const QModelIndex& )),
- this, SLOT(startQuery( const QModelIndex& )));
+ setIndentation(10);
+ header()->hide();
+ setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
+ setSortingEnabled(true);
+ sortByColumn(0, Qt::AscendingOrder);
+
+ setContextMenuPolicy(Qt::CustomContextMenu);
+ setSelectionMode(QAbstractItemView::ExtendedSelection);
+
+// // breaks with Qt 4.8
+// if(QString("4.8.0") > qVersion()) // FIXME breaks with Qt versions >= 4.10!
+ setAnimated(true);
+
+ connect(this, SIGNAL(customContextMenuRequested(const QPoint &)), SLOT(showContextMenu(const QPoint &)));
+
+#if defined Q_WS_QWS || defined Q_WS_X11
+ connect(this, SIGNAL(doubleClicked(QModelIndex)), SLOT(startQuery(QModelIndex)));
+#else
+ // afaik this is better on Mac and Windows
+ connect(this, SIGNAL(activated(QModelIndex)), SLOT(startQuery(QModelIndex)));
+#endif
}
-NickView::~NickView() {
-}
-void NickView::init() {
- if(!model())
- return;
+void NickView::init()
+{
+ if (!model())
+ return;
- for(int i = 1; i < model()->columnCount(); i++)
- setColumnHidden(i, true);
+ for (int i = 1; i < model()->columnCount(); i++)
+ setColumnHidden(i, true);
- expandAll();
+ connect(selectionModel(), SIGNAL(currentChanged(QModelIndex, QModelIndex)), SIGNAL(selectionUpdated()));
+ connect(selectionModel(), SIGNAL(selectionChanged(QItemSelection, QItemSelection)), SIGNAL(selectionUpdated()));
}
-void NickView::setModel(QAbstractItemModel *model) {
- QTreeView::setModel(model);
- init();
-}
-void NickView::rowsInserted(const QModelIndex &index, int start, int end) {
- QTreeView::rowsInserted(index, start, end);
- expandAll(); // FIXME We need to do this more intelligently. Maybe a pimped TreeView?
-}
+void NickView::setModel(QAbstractItemModel *model_)
+{
+ if (model())
+ disconnect(model(), 0, this, 0);
-QString NickView::nickFromModelIndex(const QModelIndex & index) {
- QString nick = index.sibling(index.row(), 0).data().toString();
- return nick;
+ QTreeView::setModel(model_);
+ init();
}
-BufferInfo NickView::bufferInfoFromModelIndex(const QModelIndex & index) {
- BufferInfo bufferInfo = index.data(NetworkModel::BufferInfoRole).value<BufferInfo>();
- return bufferInfo;
-}
-void NickView::showContextMenu(const QPoint & pos ) {
- QModelIndex index = indexAt(pos);
- if(index.data(NetworkModel::ItemTypeRole) != NetworkModel::IrcUserItemType) return;
+void NickView::rowsInserted(const QModelIndex &parent, int start, int end)
+{
+ QTreeView::rowsInserted(parent, start, end);
+ if (model()->data(parent, NetworkModel::ItemTypeRole) == NetworkModel::UserCategoryItemType && !isExpanded(parent)) {
+ unanimatedExpandAll();
+ }
+}
- QString nick = nickFromModelIndex(index);
- QMenu nickContextMenu(this);
+void NickView::setRootIndex(const QModelIndex &index)
+{
+ QAbstractItemView::setRootIndex(index);
+ if (index.isValid())
+ unanimatedExpandAll();
+}
- QAction *whoisAction = nickContextMenu.addAction(tr("WHOIS"));
- QAction *versionAction = nickContextMenu.addAction(tr("VERSION"));
- QAction *pingAction = nickContextMenu.addAction(tr("PING"));
- nickContextMenu.addSeparator();
+QModelIndexList NickView::selectedIndexes() const
+{
+ QModelIndexList indexList = QTreeView::selectedIndexes();
- QMenu *modeMenu = nickContextMenu.addMenu(tr("Modes"));
- QAction *opAction = modeMenu->addAction(tr("Op %1").arg(nick));
- QAction *deOpAction = modeMenu->addAction(tr("Deop %1").arg(nick));
- QAction *voiceAction = modeMenu->addAction(tr("Voice %1").arg(nick));
- QAction *deVoiceAction = modeMenu->addAction(tr("Devoice %1").arg(nick));
+ // make sure the item we clicked on is first
+ if (indexList.contains(currentIndex())) {
+ indexList.removeAll(currentIndex());
+ indexList.prepend(currentIndex());
+ }
- QMenu *kickBanMenu = nickContextMenu.addMenu(tr("Kick/Ban"));
- //TODO: add kick message from network identity (kick reason)
- QAction *kickAction = kickBanMenu->addAction(tr("Kick %1").arg(nick));
- QAction *kickBanAction = kickBanMenu->addAction(tr("Kickban %1").arg(nick));
- kickBanMenu->setEnabled(false);
- QAction *ignoreAction = nickContextMenu.addAction(tr("Ignore"));
- ignoreAction->setEnabled(false);
+ return indexList;
+}
- nickContextMenu.addSeparator();
- QAction *queryAction = nickContextMenu.addAction(tr("Query"));
- QAction *dccChatAction = nickContextMenu.addAction(tr("DCC-Chat"));
- dccChatAction->setEnabled(false);
- QAction *sendFileAction = nickContextMenu.addAction(tr("Send file"));
- sendFileAction->setEnabled(false);
+void NickView::unanimatedExpandAll()
+{
+ // since of Qt Version 4.8.0 the default expandAll will not properly work if
+ // animations are enabled. Therefore we perform an unanimated expand when a
+ // model is set or a toplevel node is inserted.
+ bool wasAnimated = isAnimated();
+ setAnimated(false);
+ expandAll();
+ setAnimated(wasAnimated);
+}
- QAction *action = nickContextMenu.exec(QCursor::pos());
- BufferInfo bufferInfo = bufferInfoFromModelIndex(index);
- if(action == whoisAction) { executeCommand(bufferInfo, QString("/WHOIS %1 %1").arg(nick)); }
- else if(action == versionAction) { executeCommand(bufferInfo, QString("/CTCP %1 VERSION").arg(nick)); }
- else if(action == pingAction) { executeCommand(bufferInfo, QString("/CTCP %1 PING ").arg(nick)); }
+void NickView::showContextMenu(const QPoint &pos)
+{
+ Q_UNUSED(pos);
- else if(action == opAction) { executeCommand(bufferInfo, QString("/OP %1").arg(nick)); }
- else if(action == deOpAction) { executeCommand(bufferInfo, QString("/DEOP %1").arg(nick)); }
- else if(action == voiceAction) { executeCommand(bufferInfo, QString("/VOICE %1").arg(nick)); }
- else if(action == deVoiceAction) { executeCommand(bufferInfo, QString("/DEVOICE %1").arg(nick)); }
+ QMenu contextMenu(this);
+ GraphicalUi::contextMenuActionProvider()->addActions(&contextMenu, selectedIndexes());
+ contextMenu.exec(QCursor::pos());
+}
- else if(action == kickAction) { executeCommand(bufferInfo, QString("/KICK %1").arg(nick)); }
- else if(action == kickBanAction) { executeCommand(bufferInfo, QString("/KICKBAN %1").arg(nick)); }
- else if(action == queryAction) { executeCommand(bufferInfo, QString("/QUERY %1").arg(nick)); }
-}
+void NickView::startQuery(const QModelIndex &index)
+{
+ if (index.data(NetworkModel::ItemTypeRole) != NetworkModel::IrcUserItemType)
+ return;
-void NickView::startQuery(const QModelIndex & index) {
- QString nick = nickFromModelIndex(index);
- BufferInfo bufferInfo = bufferInfoFromModelIndex(index);
- executeCommand(bufferInfo, QString("/QUERY %1").arg(nick));
-}
+ IrcUser *ircUser = qobject_cast<IrcUser *>(index.data(NetworkModel::IrcUserRole).value<QObject *>());
+ NetworkId networkId = index.data(NetworkModel::NetworkIdRole).value<NetworkId>();
+ if (!ircUser || !networkId.isValid())
+ return;
-void NickView::executeCommand(const BufferInfo & bufferInfo, const QString & command) {
- Client::instance()->userInput(bufferInfo, command);
+ Client::bufferModel()->switchToOrStartQuery(networkId, ircUser->nick());
}