Fixed a bug, where the customviews were not so customizable on Linux.
[quassel.git] / src / client / buffertreemodel.cpp
1 /***************************************************************************
2  *   Copyright (C) 2005-07 by The Quassel Team                             *
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) any later version.                                   *
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
21 #include <QColor>  // FIXME Dependency on QtGui!
22
23 #include "buffertreemodel.h"
24
25 #include "mappedselectionmodel.h"
26 #include <QAbstractItemView>
27
28 #include "bufferinfo.h"
29 #include "client.h"
30 #include "signalproxy.h"
31
32 /*****************************************
33 *  Fancy Buffer Items
34 *****************************************/
35 BufferTreeItem::BufferTreeItem(Buffer *buffer, TreeItem *parent) : TreeItem(parent) {
36   buf = buffer;
37   activity = Buffer::NoActivity;
38 }
39
40 uint BufferTreeItem::id() const {
41   return buf->bufferInfo().uid();
42 }
43
44 void BufferTreeItem::setActivity(const Buffer::ActivityLevel &level) {
45   activity = level;
46 }
47
48 QString BufferTreeItem::text(int column) const {
49   switch(column) {
50     case 0:
51       return buf->displayName();
52     case 1:
53       return buf->networkName();
54     default:
55       return QString();
56   }
57 }
58
59 QColor BufferTreeItem::foreground(int /*column*/) const {
60   // for the time beeing we ignore the column :)
61   if(activity & Buffer::Highlight) {
62     return QColor(Qt::red);
63   } else if(activity & Buffer::NewMessage) {
64     return QColor(Qt::darkYellow);
65   } else if(activity & Buffer::OtherActivity) {
66     return QColor(Qt::darkGreen);
67   } else {
68     if(buf->isActive())
69       return QColor(Qt::black);
70     else
71       return QColor(Qt::gray);
72   }
73 }
74
75
76 QVariant BufferTreeItem::data(int column, int role) const {
77   switch(role) {
78     case Qt::DisplayRole:
79       return text(column);
80     case Qt::ForegroundRole:
81       return foreground(column);
82     case BufferTreeModel::BufferNameRole:
83       return buf->bufferName();
84     case BufferTreeModel::BufferTypeRole:
85       return int(buf->bufferType());
86     case BufferTreeModel::BufferActiveRole:
87       return buf->isActive();
88     case BufferTreeModel::BufferUidRole:
89       return buf->bufferInfo().uid();
90     default:
91       return QVariant();
92   }
93 }
94
95 Qt::ItemFlags BufferTreeItem::flags() const {
96   Qt::ItemFlags flags = Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsDragEnabled;
97   if(buf->bufferType() == Buffer::QueryType)
98     flags |= Qt::ItemIsDropEnabled;
99
100   return flags;
101 }
102
103 /*****************************************
104 *  Network Items
105 *****************************************/
106 NetworkTreeItem::NetworkTreeItem(const QString &network, TreeItem *parent) : TreeItem(parent) {
107   net = network;
108   itemData << net << "";
109 }
110
111 uint NetworkTreeItem::id() const {
112   return qHash(net);
113 }
114
115 Qt::ItemFlags NetworkTreeItem::flags() const {
116   return Qt::ItemIsEnabled | Qt::ItemIsDropEnabled;
117 }
118
119 /*****************************************
120  * BufferTreeModel
121  *****************************************/
122 BufferTreeModel::BufferTreeModel(QObject *parent)
123   : TreeModel(BufferTreeModel::defaultHeader(), parent),
124     _selectionModelSynchronizer(new SelectionModelSynchronizer(this)),
125     _propertyMapper(new ModelPropertyMapper(this))
126 {
127   rootItem->setFlags(rootItem->flags() | Qt::ItemIsDropEnabled);
128   _propertyMapper->setModel(this);
129   delete _propertyMapper->selectionModel();
130   MappedSelectionModel *mappedSelectionModel = new MappedSelectionModel(this);
131   _propertyMapper->setSelectionModel(mappedSelectionModel);
132   synchronizeSelectionModel(mappedSelectionModel);
133   
134   connect(_selectionModelSynchronizer, SIGNAL(setCurrentIndex(QModelIndex, QItemSelectionModel::SelectionFlags)),
135           this, SLOT(setCurrentIndex(QModelIndex, QItemSelectionModel::SelectionFlags)));
136 }
137
138 QList<QVariant >BufferTreeModel::defaultHeader() {
139   QList<QVariant> data;
140   data << tr("Buffer") << tr("Network");
141   return data;
142 }
143
144 void BufferTreeModel::synchronizeSelectionModel(MappedSelectionModel *selectionModel) {
145   selectionModelSynchronizer()->addSelectionModel(selectionModel);
146 }
147
148 void BufferTreeModel::synchronizeView(QAbstractItemView *view) {
149   MappedSelectionModel *mappedSelectionModel = new MappedSelectionModel(view->model());
150   selectionModelSynchronizer()->addSelectionModel(mappedSelectionModel);
151   Q_ASSERT(mappedSelectionModel);
152   delete view->selectionModel();
153   view->setSelectionModel(mappedSelectionModel);
154 }
155
156 void BufferTreeModel::mapProperty(int column, int role, QObject *target, const QByteArray &property) {
157   propertyMapper()->addMapping(column, role, target, property);
158 }
159
160 bool BufferTreeModel::isBufferIndex(const QModelIndex &index) const {
161   // not so purdy...
162   return parent(index) != QModelIndex();
163 }
164
165 Buffer *BufferTreeModel::getBufferByIndex(const QModelIndex &index) const {
166   BufferTreeItem *item = static_cast<BufferTreeItem *>(index.internalPointer());
167   return item->buffer();
168 }
169
170 QModelIndex BufferTreeModel::getOrCreateNetworkItemIndex(Buffer *buffer) {
171   QString net = buffer->networkName();
172   TreeItem *networkItem;
173
174   if(!(networkItem = rootItem->childById(qHash(net)))) {
175     int nextRow = rootItem->childCount();
176     networkItem = new NetworkTreeItem(net, rootItem);
177     
178     beginInsertRows(QModelIndex(), nextRow, nextRow);
179     rootItem->appendChild(networkItem);
180     endInsertRows();
181   }
182
183   Q_ASSERT(networkItem);
184   return index(networkItem->row(), 0);
185 }
186
187 QModelIndex BufferTreeModel::getOrCreateBufferItemIndex(Buffer *buffer) {
188   QModelIndex networkItemIndex = getOrCreateNetworkItemIndex(buffer);
189   NetworkTreeItem *networkItem = static_cast<NetworkTreeItem*>(networkItemIndex.internalPointer());
190   TreeItem *bufferItem;
191   
192   if(!(bufferItem = networkItem->childById(buffer->bufferInfo().uid()))) {
193     int nextRow = networkItem->childCount();
194     bufferItem = new BufferTreeItem(buffer, networkItem);
195     
196     beginInsertRows(networkItemIndex, nextRow, nextRow);
197     networkItem->appendChild(bufferItem);
198     endInsertRows();
199   }
200
201   Q_ASSERT(bufferItem);
202   return index(bufferItem->row(), 0, networkItemIndex);
203 }
204
205 QStringList BufferTreeModel::mimeTypes() const {
206   QStringList types;
207   types << "application/Quassel/BufferItem/row"
208     << "application/Quassel/BufferItem/network"
209     << "application/Quassel/BufferItem/bufferInfo";
210   return types;
211 }
212
213 QMimeData *BufferTreeModel::mimeData(const QModelIndexList &indexes) const {
214   QMimeData *mimeData = new QMimeData();
215
216   QModelIndex index = indexes.first();
217   
218   mimeData->setData("application/Quassel/BufferItem/row", QByteArray::number(index.row()));
219   mimeData->setData("application/Quassel/BufferItem/network", getBufferByIndex(index)->networkName().toUtf8());
220   mimeData->setData("application/Quassel/BufferItem/bufferInfo", QByteArray::number(getBufferByIndex(index)->bufferInfo().uid()));
221   return mimeData;
222 }
223
224 bool BufferTreeModel::dropMimeData(const QMimeData *data, Qt::DropAction /*action*/, int /*row*/, int /*column*/, const QModelIndex &parent) {
225   foreach(QString mimeType, mimeTypes()) {
226     if(!(data->hasFormat(mimeType)))
227       return false; // whatever the drop is... it's not a buffer...
228   }
229   
230   int sourcerow = data->data("application/Quassel/BufferItem/row").toInt();
231   QString network = QString::fromUtf8(data->data("application/Quassel/BufferItem/network"));
232   
233   Q_ASSERT(rootItem->childById(qHash(network)));
234
235   if(parent == QModelIndex()) // can't be a query...
236     return false;
237   
238   Buffer *sourceBuffer = static_cast<BufferTreeItem *>(rootItem->childById(qHash(network))->child(sourcerow))->buffer();
239   Buffer *targetBuffer = getBufferByIndex(parent);
240
241   if(!(sourceBuffer->bufferType() & targetBuffer->bufferType() & Buffer::QueryType)) // only queries can be merged
242     return false;
243   
244   if(sourceBuffer == targetBuffer) // we won't merge with ourself :)
245     return false;
246     
247   // TODO: warn user about buffermerge!
248   qDebug() << "merging" << sourceBuffer->bufferName() << "with" << targetBuffer->bufferName();
249   removeRow(parent.row(), BufferTreeModel::parent(parent));
250   
251   return true;
252 }
253
254 void BufferTreeModel::bufferUpdated(Buffer *buffer) {
255   QModelIndex itemindex = getOrCreateBufferItemIndex(buffer);
256   emit invalidateFilter();
257   emit dataChanged(itemindex, itemindex);
258 }
259
260 // This Slot indicates that the user has selected a different buffer in the gui
261 void BufferTreeModel::setCurrentIndex(const QModelIndex &index, QItemSelectionModel::SelectionFlags command) {
262   Q_UNUSED(command)
263   if(isBufferIndex(index)) {
264     currentBuffer = getBufferByIndex(index);
265     bufferActivity(Buffer::NoActivity, currentBuffer);
266     emit bufferSelected(currentBuffer);
267     emit selectionChanged(index);
268   }
269 }
270
271 void BufferTreeModel::bufferActivity(Buffer::ActivityLevel level, Buffer *buffer) {
272   BufferTreeItem *bufferItem = static_cast<BufferTreeItem*>(getOrCreateBufferItemIndex(buffer).internalPointer());
273   if(buffer != currentBuffer)
274     bufferItem->setActivity(level);
275   else
276     bufferItem->setActivity(Buffer::NoActivity);
277   bufferUpdated(buffer);
278 }
279
280 void BufferTreeModel::selectBuffer(Buffer *buffer) {
281   QModelIndex index = getOrCreateBufferItemIndex(buffer);
282   // SUPER UGLY!
283   setCurrentIndex(index, 0);
284 }