Fixing BR #335 - needs a core restart
[quassel.git] / src / qtui / settingspages / aliasesmodel.cpp
1 /***************************************************************************
2  *   Copyright (C) 2005-08 by the Quassel Project                          *
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) version 3.                                           *
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 "aliasesmodel.h"
22
23 #include <QDebug>
24 #include <QStringList>
25
26 #include "client.h"
27 #include "signalproxy.h"
28
29 AliasesModel::AliasesModel(QObject *parent)
30   : QAbstractItemModel(parent),
31     _configChanged(false)
32 {
33   // we need this signal for future connects to reset the data;
34   connect(Client::instance(), SIGNAL(connected()), this, SLOT(clientConnected()));
35   connect(Client::instance(), SIGNAL(disconnected()), this, SLOT(clientDisconnected()));
36   if(Client::isConnected())
37     clientConnected();
38   else
39     emit modelReady(false);
40 }
41
42 QVariant AliasesModel::data(const QModelIndex &index, int role) const {
43   if(!index.isValid() || index.row() >= rowCount() || index.column() >= columnCount())
44     return QVariant();
45
46   switch(role) {
47   case Qt::ToolTipRole:
48     switch(index.column()) {
49     case 0:
50       return "<b>The shortcut for the alias</b><br />"
51         "It can be used as a regular slash command.<br /><br />"
52         "<b>Example:</b> \"foo\" can be used per /foo";
53     case 1:
54       return "<b>The string the shortcut will be expanded to</b><br />"
55         "<b>special variables:</b><br />"
56         " - <b>$i</b> represenents the i'th parameter.<br />"
57         " - <b>$i..j</b> represenents the i'th to j'th parameter separated by spaces.<br />"
58         " - <b>$i..</b> represenents all parameters from i on separated by spaces.<br />"
59         " - <b>$i:hostname</b> represents the hostname of the user identified by the i'th parameter or a * if unknown.<br />"
60         " - <b>$0</b> the whole string.<br />"
61         " - <b>$currentnick</b> your current nickname<br />"
62         " - <b>$channelname</b> the name of the selected channel<br /><br />"
63         "Multiple commands can be separated with semicolons<br /><br />"
64         "<b>Example:</b> \"Test $1; Test $2; Test All $0\" will be expanded to three separate messages \"Test 1\", \"Test 2\" and \"Test All 1 2 3\" when called like /test 1 2 3";
65     default:
66       return QVariant();
67     }
68   case Qt::DisplayRole:
69   case Qt::EditRole:
70     switch(index.column()) {
71     case 0:
72       return aliasManager()[index.row()].name;
73     case 1:
74       return aliasManager()[index.row()].expansion;
75     default:
76       return QVariant();
77     }
78   default:
79     return QVariant();
80   }
81 }
82
83 bool AliasesModel::setData(const QModelIndex &index, const QVariant &value, int role) {
84   if(!index.isValid() || index.row() >= rowCount() || index.column() >= columnCount() || role != Qt::EditRole)
85     return false;
86
87   QString newValue = value.toString();
88   if(newValue.isEmpty())
89     return false;
90   
91   switch(index.column()) {
92   case 0:
93     if(aliasManager().contains(newValue)) {
94       return false;
95     } else {
96       cloneAliasManager()[index.row()].name = newValue;
97       return true;
98     }
99   case 1:
100     cloneAliasManager()[index.row()].expansion = newValue;
101     return true;
102   default:
103     return false;
104   }
105 }
106
107 void AliasesModel::newAlias() {
108   QString newName("alias");
109   int i = 0;
110   AliasManager &manager = cloneAliasManager();
111   while(manager.contains(newName)) {
112     i++;
113     newName = QString("alias%1").arg(i);
114   }
115   beginInsertRows(QModelIndex(), rowCount(), rowCount());
116   manager.addAlias(newName, "Expansion");
117   endInsertRows();
118 }
119
120 void AliasesModel::removeAlias(int index) {
121   if(index < 0 || index >= rowCount())
122     return;
123
124   AliasManager &manager = cloneAliasManager();  
125   beginRemoveRows(QModelIndex(), index, index);
126   manager.removeAt(index);
127   endRemoveRows();
128 }
129
130 Qt::ItemFlags AliasesModel::flags(const QModelIndex &index) const {
131   if(!index.isValid()) {
132     return Qt::ItemIsDropEnabled;
133   } else {
134     return Qt::ItemIsSelectable | Qt::ItemIsEnabled | Qt::ItemIsEditable;
135   }
136 }
137
138
139 QVariant AliasesModel::headerData(int section, Qt::Orientation orientation, int role) const {
140   QStringList header;
141   header << tr("Alias")
142          << tr("Expansion");
143   
144   if(orientation == Qt::Horizontal && role == Qt::DisplayRole)
145     return header[section];
146
147   return QVariant();
148 }
149
150 QModelIndex AliasesModel::index(int row, int column, const QModelIndex &parent) const {
151   Q_UNUSED(parent);
152   if(row >= rowCount() || column >= columnCount())
153     return QModelIndex();
154
155   return createIndex(row, column);
156 }
157
158
159 const AliasManager &AliasesModel::aliasManager() const {
160   if(_configChanged)
161     return _clonedAliasManager;
162   else
163     return _aliasManager;
164 }
165
166 AliasManager &AliasesModel::aliasManager() {
167   if(_configChanged)
168     return _clonedAliasManager;
169   else
170     return _aliasManager;
171 }
172
173 AliasManager &AliasesModel::cloneAliasManager() {
174   if(!_configChanged) {
175     _clonedAliasManager = _aliasManager;
176     _configChanged = true;
177     emit configChanged(true);
178   }
179   return _clonedAliasManager;
180 }
181
182 void AliasesModel::revert() {
183   if(!_configChanged)
184     return;
185   
186   _configChanged = false;
187   emit configChanged(false);
188   reset();
189 }
190
191 void AliasesModel::commit() {
192   if(!_configChanged)
193     return;
194
195   _aliasManager.requestUpdate(_clonedAliasManager.toVariantMap());
196   revert();
197 }  
198
199 void AliasesModel::initDone() {
200   reset();
201   emit modelReady(true);
202 }
203
204 void AliasesModel::clientConnected() {
205   _aliasManager = AliasManager();
206   Client::signalProxy()->synchronize(&_aliasManager);
207   connect(&_aliasManager, SIGNAL(initDone()), this, SLOT(initDone()));
208   connect(&_aliasManager, SIGNAL(updated(const QVariantMap &)), this, SLOT(revert()));
209 }
210
211 void AliasesModel::clientDisconnected() {
212   // clear alias managers
213   _aliasManager = AliasManager();
214   _clonedAliasManager = AliasManager();
215   reset();
216   emit modelReady(false);
217 }