Reorganizing of the Quassel architecture is almost done. Client and GUI have been...
[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 "global.h"
24 #include "buffertreemodel.h"
25
26 /*****************************************
27 *  Fancy Buffer Items
28 *****************************************/
29 BufferTreeItem::BufferTreeItem(Buffer *buffer, TreeItem *parent) : TreeItem(parent) {
30   buf = buffer;
31   activity = Buffer::NoActivity;
32 }
33
34 void BufferTreeItem::setActivity(const Buffer::ActivityLevel &level) {
35   activity = level;
36 }
37
38 QString BufferTreeItem::text(int column) const {
39   switch(column) {
40     case 0:
41       return buf->displayName();
42     case 1:
43       return buf->networkName();
44     default:
45       return QString();
46   }
47 }
48
49 QColor BufferTreeItem::foreground(int column) const {
50   // for the time beeing we ignore the column :)
51   if(activity & Buffer::Highlight) {
52     return QColor(Qt::red);
53   } else if(activity & Buffer::NewMessage) {
54     return QColor(Qt::darkYellow);
55   } else if(activity & Buffer::OtherActivity) {
56     return QColor(Qt::darkGreen);
57   } else {
58     if(buf->isActive())
59       return QColor(Qt::black);
60     else
61       return QColor(Qt::gray);
62   }
63 }
64
65
66 QVariant BufferTreeItem::data(int column, int role) const {
67   switch(role) {
68     case Qt::DisplayRole:
69       return text(column);
70     case Qt::ForegroundRole:
71       return foreground(column);
72     case BufferTreeModel::BufferTypeRole:
73       return buf->bufferType();
74     case BufferTreeModel::BufferActiveRole:
75       return buf->isActive();
76     default:
77       return QVariant();
78   }
79 }
80
81 /*****************************************
82  * BufferTreeModel
83  *****************************************/
84 BufferTreeModel::BufferTreeModel(QObject *parent) : TreeModel(BufferTreeModel::defaultHeader(), parent) {
85   connect(this, SIGNAL(fakeUserInput(BufferId, QString)), ClientProxy::instance(), SLOT(gsUserInput(BufferId, QString)));
86 }
87
88 QList<QVariant >BufferTreeModel::defaultHeader() {
89   QList<QVariant> data;
90   data << tr("Buffer") << tr("Network");
91   return data;
92 }
93
94
95 Qt::ItemFlags BufferTreeModel::flags(const QModelIndex &index) const {
96   if(!index.isValid())
97     return 0;
98
99   // I think this is pretty ugly..
100   if(isBufferIndex(index)) {
101     Buffer *buffer = getBufferByIndex(index);
102     if(buffer->bufferType() == Buffer::QueryBuffer)
103       return Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsDragEnabled | Qt::ItemIsDropEnabled;
104     else
105       return Qt::ItemIsEnabled | Qt::ItemIsSelectable;
106   } else {
107     return Qt::ItemIsEnabled | Qt::ItemIsDropEnabled; 
108   }
109 }
110
111 bool BufferTreeModel::isBufferIndex(const QModelIndex &index) const {
112   return parent(index) != QModelIndex();
113 }
114
115 Buffer *BufferTreeModel::getBufferByIndex(const QModelIndex &index) const {
116   BufferTreeItem *item = static_cast<BufferTreeItem *>(index.internalPointer());
117   return item->buffer();
118 }
119
120 QModelIndex BufferTreeModel::getOrCreateNetworkItemIndex(Buffer *buffer) {
121   QString net = buffer->networkName();
122   
123   if(networkItem.contains(net)) {
124     return index(networkItem[net]->row(), 0);
125   } else {
126     QList<QVariant> data;
127     data << net << "";
128     
129     int nextRow = rootItem->childCount();
130     
131     beginInsertRows(QModelIndex(), nextRow, nextRow);
132     rootItem->appendChild(new TreeItem(data, rootItem));
133     endInsertRows();
134     
135     networkItem[net] = rootItem->child(nextRow);
136     return index(nextRow, 0);
137   }
138 }
139
140 QModelIndex BufferTreeModel::getOrCreateBufferItemIndex(Buffer *buffer) {
141   QModelIndex networkItemIndex = getOrCreateNetworkItemIndex(buffer);
142   
143   if(bufferItem.contains(buffer)) {
144     return index(bufferItem[buffer]->row(), 0, networkItemIndex);
145   } else {
146     // first we determine the parent of the new Item
147     TreeItem *networkItem = static_cast<TreeItem*>(networkItemIndex.internalPointer());
148
149     int nextRow = networkItem->childCount();
150
151     beginInsertRows(networkItemIndex, nextRow, nextRow);
152     networkItem->appendChild(new BufferTreeItem(buffer, networkItem));
153     endInsertRows();
154
155     bufferItem[buffer] = static_cast<BufferTreeItem *>(networkItem->child(nextRow));
156     return index(nextRow, 0, networkItemIndex);
157   }
158 }
159
160 QStringList BufferTreeModel::mimeTypes() const {
161   QStringList types;
162   types << "application/Quassel/BufferItem/row"
163     << "application/Quassel/BufferItem/network";
164   return types;
165 }
166
167 QMimeData *BufferTreeModel::mimeData(const QModelIndexList &indexes) const {
168   QMimeData *mimeData = new QMimeData();
169
170   QModelIndex index = indexes.first();
171   
172   mimeData->setData("application/Quassel/BufferItem/row", QByteArray::number(index.row()));
173   mimeData->setData("application/Quassel/BufferItem/network", getBufferByIndex(index)->networkName().toUtf8());
174   return mimeData;
175 }
176
177 bool BufferTreeModel::dropMimeData(const QMimeData *data, Qt::DropAction action, int row, int column, const QModelIndex &parent) {
178   int sourcerow = data->data("application/Quassel/BufferItem/row").toInt();
179   QString network = QString::fromUtf8(data->data("application/Quassel/BufferItem/network"));
180   
181   if(!networkItem.contains(network))
182     return false;
183
184   if(!isBufferIndex(parent)) // dropping at a network -> no merging needed
185     return false;
186
187   Buffer *sourceBuffer = static_cast<BufferTreeItem *>(networkItem[network]->child(sourcerow))->buffer();
188   Buffer *targetBuffer = getBufferByIndex(parent);
189   
190   if(sourceBuffer == targetBuffer) // we won't merge with ourself :)
191     return false;
192   
193   /*
194   if(QMessageBox::warning(static_cast<QWidget *>(QObject::parent()),
195                           tr("Merge Buffers?"),
196                           tr("Do you really want to merge the following Buffers?<br />%1.%2<br />%3.%4").arg(sourceBuffer->networkName()).arg(sourceBuffer->bufferName()).arg(targetBuffer->networkName()).arg(targetBuffer->bufferName()),
197                           QMessageBox::Yes|QMessageBox::No) == QMessageBox::No)
198     return false;
199
200   */
201   qDebug() << "merging" << sourceBuffer->bufferName() << "with" << targetBuffer->bufferName();
202   bufferItem.remove(getBufferByIndex(parent));
203   removeRow(parent.row(), BufferTreeModel::parent(parent));
204   
205   return true;
206 }
207
208 void BufferTreeModel::bufferUpdated(Buffer *buffer) {
209   QModelIndex itemindex = getOrCreateBufferItemIndex(buffer);
210   emit invalidateFilter();
211   emit dataChanged(itemindex, itemindex);
212 }
213
214 // This Slot indicates that the user has selected a different buffer in the gui
215 void BufferTreeModel::changeCurrent(const QModelIndex &current, const QModelIndex &previous) {
216   if(isBufferIndex(current)) {
217     currentBuffer = getBufferByIndex(current);
218     bufferActivity(Buffer::NoActivity, currentBuffer);
219     emit bufferSelected(currentBuffer);
220     emit updateSelection(current, QItemSelectionModel::ClearAndSelect);
221   }
222 }
223
224 // we received a double click on a buffer, so we're going to join it
225 void BufferTreeModel::doubleClickReceived(const QModelIndex &clicked) {
226   if(isBufferIndex(clicked)) {
227     Buffer *buffer = getBufferByIndex(clicked);
228     if(!buffer->isStatusBuffer()) 
229       emit fakeUserInput(buffer->bufferId(), QString("/join " + buffer->bufferName()));
230   }
231     
232 }
233
234 void BufferTreeModel::bufferActivity(Buffer::ActivityLevel level, Buffer *buffer) {
235   if(bufferItem.contains(buffer) and buffer != currentBuffer)
236     bufferItem[buffer]->setActivity(level);
237   else
238     bufferItem[buffer]->setActivity(Buffer::NoActivity);
239   bufferUpdated(buffer);
240 }
241
242 void BufferTreeModel::selectBuffer(Buffer *buffer) {
243   QModelIndex index = getOrCreateBufferItemIndex(buffer);
244   emit updateSelection(index, QItemSelectionModel::ClearAndSelect);
245 }