Fix default button behavior in CoreConnectDlg.
[quassel.git] / src / qtui / coreconnectdlg.cpp
1 /***************************************************************************
2  *   Copyright (C) 2005-08 by the Quassel IRC 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) 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 <QDebug>
22 #include <QMessageBox>
23
24 #include "coreconnectdlg.h"
25
26 #include "clientsettings.h"
27 #include "clientsyncer.h"
28 #include "coreconfigwizard.h"
29
30 CoreConnectDlg::CoreConnectDlg(QWidget *parent, bool autoconnect) : QDialog(parent) {
31   ui.setupUi(this);
32
33   clientSyncer = new ClientSyncer(this);
34   wizard = 0;
35
36   setAttribute(Qt::WA_DeleteOnClose);
37
38   doingAutoConnect = false;
39
40   ui.stackedWidget->setCurrentWidget(ui.accountPage);
41
42   CoreAccountSettings s;
43   AccountId lastacc = s.lastAccount();
44   autoConnectAccount = s.autoConnectAccount();
45   QListWidgetItem *currentItem = 0;
46   foreach(AccountId id, s.knownAccounts()) {
47     if(!id.isValid()) continue;
48     QVariantMap data = s.retrieveAccountData(id);
49     accounts[id] = data;
50     QListWidgetItem *item = new QListWidgetItem(data["AccountName"].toString(), ui.accountList);
51     item->setData(Qt::UserRole, QVariant::fromValue<AccountId>(id));
52     if(id == lastacc) currentItem = item;
53   }
54   if(currentItem) ui.accountList->setCurrentItem(currentItem);
55   else ui.accountList->setCurrentRow(0);
56
57   setAccountWidgetStates();
58
59   ui.accountButtonBox->button(QDialogButtonBox::Ok)->setFocus();
60
61   connect(clientSyncer, SIGNAL(socketStateChanged(QAbstractSocket::SocketState)),this, SLOT(initPhaseSocketState(QAbstractSocket::SocketState)));
62   connect(clientSyncer, SIGNAL(connectionError(const QString &)), this, SLOT(initPhaseError(const QString &)));
63   connect(clientSyncer, SIGNAL(connectionMsg(const QString &)), this, SLOT(initPhaseMsg(const QString &)));
64   connect(clientSyncer, SIGNAL(startLogin()), this, SLOT(startLogin()));
65   connect(clientSyncer, SIGNAL(loginFailed(const QString &)), this, SLOT(loginFailed(const QString &)));
66   connect(clientSyncer, SIGNAL(loginSuccess()), this, SLOT(startSync()));
67   connect(clientSyncer, SIGNAL(startCoreSetup(const QVariantList &)), this, SLOT(startCoreConfig(const QVariantList &)));
68   connect(clientSyncer, SIGNAL(sessionProgress(quint32, quint32)), this, SLOT(coreSessionProgress(quint32, quint32)));
69   connect(clientSyncer, SIGNAL(networksProgress(quint32, quint32)), this, SLOT(coreNetworksProgress(quint32, quint32)));
70   connect(clientSyncer, SIGNAL(channelsProgress(quint32, quint32)), this, SLOT(coreChannelsProgress(quint32, quint32)));
71   connect(clientSyncer, SIGNAL(ircUsersProgress(quint32, quint32)), this, SLOT(coreIrcUsersProgress(quint32, quint32)));
72   connect(clientSyncer, SIGNAL(syncFinished()), this, SLOT(syncFinished()));
73
74   connect(ui.user, SIGNAL(textChanged(const QString &)), this, SLOT(setLoginWidgetStates()));
75   connect(ui.password, SIGNAL(textChanged(const QString &)), this, SLOT(setLoginWidgetStates()));
76
77   connect(ui.loginButtonBox, SIGNAL(rejected()), this, SLOT(restartPhaseNull()));
78   connect(ui.syncButtonBox->button(QDialogButtonBox::Abort), SIGNAL(clicked()), this, SLOT(restartPhaseNull()));
79
80   if(autoconnect && ui.accountList->count() && autoConnectAccount.isValid()
81      && autoConnectAccount == ui.accountList->currentItem()->data(Qt::UserRole).value<AccountId>()) {
82     doingAutoConnect = true;
83     on_accountButtonBox_accepted();
84   }
85 }
86
87 CoreConnectDlg::~CoreConnectDlg() {
88   if(ui.accountList->selectedItems().count()) {
89     CoreAccountSettings s;
90     s.setLastAccount(ui.accountList->selectedItems()[0]->data(Qt::UserRole).value<AccountId>());
91   }
92 }
93
94
95 /****************************************************
96  * Account Management
97  ***************************************************/
98
99 void CoreConnectDlg::on_accountList_itemSelectionChanged() {
100   setAccountWidgetStates();
101 }
102
103 void CoreConnectDlg::setAccountWidgetStates() {
104   QList<QListWidgetItem *> selectedItems = ui.accountList->selectedItems();
105   ui.editAccount->setEnabled(selectedItems.count());
106   ui.deleteAccount->setEnabled(selectedItems.count());
107   ui.autoConnect->setEnabled(selectedItems.count());
108   if(selectedItems.count()) {
109     ui.autoConnect->setChecked(selectedItems[0]->data(Qt::UserRole).value<AccountId>() == autoConnectAccount);
110   }
111   ui.accountButtonBox->button(QDialogButtonBox::Ok)->setEnabled(ui.accountList->count());
112   //ui.accountButtonBox->button(QDialogButtonBox::Ok)->setDefault(true);
113   //ui.accountButtonBox->button(QDialogButtonBox::Ok)->setFocus();
114 }
115
116 void CoreConnectDlg::on_autoConnect_clicked(bool state) {
117   if(!state) {
118     autoConnectAccount = 0;
119   } else {
120     if(ui.accountList->selectedItems().count()) {
121       autoConnectAccount = ui.accountList->selectedItems()[0]->data(Qt::UserRole).value<AccountId>();
122     } else {
123       qWarning() << "Checked auto connect without an enabled item!";  // should never happen!
124       autoConnectAccount = 0;
125     }
126   }
127   setAccountWidgetStates();
128 }
129
130 void CoreConnectDlg::on_addAccount_clicked() {
131   QStringList existing;
132   for(int i = 0; i < ui.accountList->count(); i++) existing << ui.accountList->item(i)->text();
133   CoreAccountEditDlg dlg(0, QVariantMap(), existing, this);
134   if(dlg.exec() == QDialog::Accepted) {
135     // find free ID
136     AccountId id = accounts.count() + 1;
137     for(AccountId i = 1; i <= accounts.count(); i++) {
138       if(!accounts.keys().contains(i)) {
139         id = i;
140         break;
141       }
142     }
143     QVariantMap data = dlg.accountData();
144     data["AccountId"] = QVariant::fromValue<AccountId>(id);
145     accounts[id] = data;
146     QListWidgetItem *item = new QListWidgetItem(data["AccountName"].toString(), ui.accountList);
147     item->setData(Qt::UserRole, QVariant::fromValue<AccountId>(id));
148     ui.accountList->setCurrentItem(item);
149   }
150 }
151
152 void CoreConnectDlg::on_editAccount_clicked() {
153   QStringList existing;
154   for(int i = 0; i < ui.accountList->count(); i++) existing << ui.accountList->item(i)->text();
155   AccountId id = ui.accountList->currentItem()->data(Qt::UserRole).value<AccountId>();
156   QVariantMap acct = accounts[id];
157   CoreAccountEditDlg dlg(id, acct, existing, this);
158   if(dlg.exec() == QDialog::Accepted) {
159     QVariantMap data = dlg.accountData();
160     ui.accountList->currentItem()->setText(data["AccountName"].toString());
161     accounts[id] = data;
162   }
163 }
164
165 void CoreConnectDlg::on_deleteAccount_clicked() {
166   AccountId id = ui.accountList->currentItem()->data(Qt::UserRole).value<AccountId>();
167   int ret = QMessageBox::question(this, tr("Remove Account Settings"),
168                                   tr("Do you really want to remove your local settings for this Quassel Core account?<br>"
169                                   "Note: This will <em>not</em> remove or change any data on the Core itself!"),
170                                   QMessageBox::Yes|QMessageBox::No, QMessageBox::No);
171   if(ret == QMessageBox::Yes) {
172     int idx = ui.accountList->currentRow();
173     delete ui.accountList->takeItem(idx);
174     ui.accountList->setCurrentRow(qMin(idx, ui.accountList->count()-1));
175     accounts[id]["Delete"] = true;  // we only flag this here, actual deletion happens on accept!
176     setAccountWidgetStates();
177   }
178 }
179
180 void CoreConnectDlg::on_accountList_itemDoubleClicked(QListWidgetItem *item) {
181   Q_UNUSED(item);
182   on_accountButtonBox_accepted();
183 }
184
185 void CoreConnectDlg::on_accountButtonBox_accepted() {
186   // save accounts
187   CoreAccountSettings s;
188   foreach(QVariantMap acct, accounts.values()) {
189     AccountId id = acct["AccountId"].value<AccountId>();
190     if(acct.contains("Delete")) {
191       s.removeAccount(id);
192     } else {
193       s.storeAccountData(id, acct);
194     }
195   }
196   s.setAutoConnectAccount(autoConnectAccount);
197
198   ui.stackedWidget->setCurrentWidget(ui.loginPage);
199   account = ui.accountList->currentItem()->data(Qt::UserRole).value<AccountId>();
200   accountData = accounts[account];
201   s.setLastAccount(account);
202   connectToCore();
203 }
204
205 /*****************************************************
206  * Connecting to the Core
207  ****************************************************/
208
209 /*** Phase One: initializing the core connection ***/
210
211 void CoreConnectDlg::connectToCore() {
212   ui.connectIcon->setPixmap(QPixmap::fromImage(QImage(":/22x22/actions/network-disconnect")));
213   ui.connectLabel->setText(tr("Connect to %1").arg(accountData["Host"].toString()));
214   ui.coreInfoLabel->setText("");
215   ui.loginStack->setCurrentWidget(ui.loginEmptyPage);
216   ui.loginButtonBox->setStandardButtons(QDialogButtonBox::Cancel|QDialogButtonBox::Ok);
217   ui.loginButtonBox->button(QDialogButtonBox::Ok)->setDefault(true);
218   ui.loginButtonBox->button(QDialogButtonBox::Ok)->setDisabled(true);
219   disconnect(ui.loginButtonBox, 0, this, 0);
220   connect(ui.loginButtonBox, SIGNAL(rejected()), this, SLOT(restartPhaseNull()));
221
222   clientSyncer->connectToCore(accountData);
223 }
224
225 void CoreConnectDlg::initPhaseError(const QString &error) {
226   doingAutoConnect = false;
227   ui.connectIcon->setPixmap(QPixmap::fromImage(QImage(":/22x22/status/dialog-error")));
228   //ui.connectLabel->setBrush(QBrush("red"));
229   ui.connectLabel->setText(tr("<div style=color:red;>Connection to %1 failed!</div>").arg(accountData["Host"].toString()));
230   ui.coreInfoLabel->setText(error);
231   ui.loginButtonBox->setStandardButtons(QDialogButtonBox::Retry|QDialogButtonBox::Cancel);
232   //ui.loginButtonBox->button(QDialogButtonBox::Retry)->setDefault(true);
233   ui.loginButtonBox->button(QDialogButtonBox::Retry)->setFocus();
234   disconnect(ui.loginButtonBox, 0, this, 0);
235   connect(ui.loginButtonBox, SIGNAL(accepted()), this, SLOT(restartPhaseNull()));
236   connect(ui.loginButtonBox, SIGNAL(rejected()), this, SLOT(reject()));
237 }
238
239 void CoreConnectDlg::initPhaseMsg(const QString &msg) {
240   ui.coreInfoLabel->setText(msg);
241 }
242
243 void CoreConnectDlg::initPhaseSocketState(QAbstractSocket::SocketState state) {
244   QString s;
245   QString host = accountData["Host"].toString();
246   switch(state) {
247     case QAbstractSocket::UnconnectedState: s = tr("Not connected to %1.").arg(host); break;
248     case QAbstractSocket::HostLookupState: s = tr("Looking up %1...").arg(host); break;
249     case QAbstractSocket::ConnectingState: s = tr("Connecting to %1...").arg(host); break;
250     case QAbstractSocket::ConnectedState: s = tr("Connected to %1").arg(host); break;
251     default: s = tr("Unknown connection state to %1"); break;
252   }
253   ui.connectLabel->setText(s);
254 }
255
256 void CoreConnectDlg::restartPhaseNull() {
257   doingAutoConnect = false;
258   ui.stackedWidget->setCurrentWidget(ui.accountPage);
259   clientSyncer->disconnectFromCore();
260 }
261
262 /*********************************************************
263  * Phase Two: Login
264  *********************************************************/
265
266 void CoreConnectDlg::startLogin() {
267   ui.connectIcon->setPixmap(QPixmap::fromImage(QImage(":/22x22/actions/network-connect")));
268   ui.loginStack->setCurrentWidget(ui.loginCredentialsPage);
269   //ui.loginStack->setMinimumSize(ui.loginStack->sizeHint()); ui.loginStack->updateGeometry();
270   ui.loginButtonBox->setStandardButtons(QDialogButtonBox::Ok|QDialogButtonBox::Cancel);
271   ui.loginButtonBox->button(QDialogButtonBox::Ok)->setDefault(true);
272   ui.loginButtonBox->button(QDialogButtonBox::Ok)->setFocus();
273   if(!accountData["User"].toString().isEmpty()) {
274     ui.user->setText(accountData["User"].toString());
275     if(accountData["RememberPasswd"].toBool()) {
276       ui.password->setText(accountData["Password"].toString());
277       ui.rememberPasswd->setChecked(true);
278       ui.loginButtonBox->button(QDialogButtonBox::Ok)->setFocus();
279     } else {
280       ui.rememberPasswd->setChecked(false);
281       ui.password->setFocus();
282     }
283   } else ui.user->setFocus();
284   disconnect(ui.loginButtonBox, 0, this, 0);
285   connect(ui.loginButtonBox, SIGNAL(accepted()), this, SLOT(doLogin()));
286   connect(ui.loginButtonBox, SIGNAL(rejected()), this, SLOT(restartPhaseNull()));
287   if(doingAutoConnect) doLogin();
288 }
289
290 void CoreConnectDlg::doLogin() {
291   QVariantMap loginData;
292   loginData["User"] = ui.user->text();
293   loginData["Password"] = ui.password->text();
294   loginData["RememberPasswd"] = ui.rememberPasswd->isChecked();
295   doLogin(loginData);
296 }
297
298 void CoreConnectDlg::doLogin(const QVariantMap &loginData) {
299   disconnect(ui.loginButtonBox, 0, this, 0);
300   connect(ui.loginButtonBox, SIGNAL(accepted()), this, SLOT(doLogin()));
301   connect(ui.loginButtonBox, SIGNAL(rejected()), this, SLOT(restartPhaseNull()));
302   ui.loginStack->setCurrentWidget(ui.loginCredentialsPage);
303   ui.loginGroup->setTitle(tr("Logging in..."));
304   ui.user->setDisabled(true);
305   ui.password->setDisabled(true);
306   ui.rememberPasswd->setDisabled(true);
307   ui.loginButtonBox->button(QDialogButtonBox::Ok)->setDisabled(true);
308   accountData["User"] = loginData["User"];
309   accountData["RememberPasswd"] = loginData["RememberPasswd"];
310   if(loginData["RememberPasswd"].toBool()) accountData["Password"] = loginData["Password"];
311   else accountData.remove("Password");
312   ui.user->setText(loginData["User"].toString());
313   ui.password->setText(loginData["Password"].toString());
314   ui.rememberPasswd->setChecked(loginData["RememberPasswd"].toBool());
315   CoreAccountSettings s;
316   s.storeAccountData(account, accountData);
317   clientSyncer->loginToCore(loginData["User"].toString(), loginData["Password"].toString());
318 }
319
320 void CoreConnectDlg::setLoginWidgetStates() {
321   ui.loginButtonBox->button(QDialogButtonBox::Ok)->setDisabled(ui.user->text().isEmpty() || ui.password->text().isEmpty());
322 }
323
324 void CoreConnectDlg::loginFailed(const QString &error) {
325   if(wizard) {
326     wizard->reject();
327   }
328   ui.loginStack->setCurrentWidget(ui.loginCredentialsPage);
329   ui.loginGroup->setTitle(tr("Login"));
330   ui.user->setEnabled(true);
331   ui.password->setEnabled(true);
332   ui.rememberPasswd->setEnabled(true);
333   ui.coreInfoLabel->setText(error);
334   ui.loginButtonBox->button(QDialogButtonBox::Ok)->setEnabled(true);
335   ui.password->setFocus();
336   doingAutoConnect = false;
337 }
338
339 void CoreConnectDlg::startCoreConfig(const QVariantList &backends) {
340   storageBackends = backends;
341   ui.loginStack->setCurrentWidget(ui.coreConfigPage);
342
343   //on_launchCoreConfigWizard_clicked();
344
345 }
346
347 void CoreConnectDlg::on_launchCoreConfigWizard_clicked() {
348   Q_ASSERT(!wizard);
349   wizard = new CoreConfigWizard(storageBackends, this);
350   connect(wizard, SIGNAL(setupCore(const QVariant &)), clientSyncer, SLOT(doCoreSetup(const QVariant &)));
351   connect(wizard, SIGNAL(loginToCore(const QVariantMap &)), this, SLOT(doLogin(const QVariantMap &)));
352   connect(clientSyncer, SIGNAL(coreSetupSuccess()), wizard, SLOT(coreSetupSuccess()));
353   connect(clientSyncer, SIGNAL(coreSetupFailed(const QString &)), wizard, SLOT(coreSetupFailed(const QString &)));
354   connect(wizard, SIGNAL(accepted()), this, SLOT(configWizardAccepted()));
355   connect(wizard, SIGNAL(rejected()), this, SLOT(configWizardRejected()));
356   connect(clientSyncer, SIGNAL(loginSuccess()), wizard, SLOT(loginSuccess()));
357   connect(clientSyncer, SIGNAL(syncFinished()), wizard, SLOT(syncFinished()));
358   wizard->show();
359 }
360
361 void CoreConnectDlg::configWizardAccepted() {
362
363   wizard->deleteLater();
364   wizard = 0;
365 }
366
367 void CoreConnectDlg::configWizardRejected() {
368
369   wizard->deleteLater();
370   wizard = 0;
371   //exit(1); // FIXME
372 }
373
374
375 /************************************************************
376  * Phase Three: Syncing
377  ************************************************************/
378
379 void CoreConnectDlg::startSync() {
380   ui.sessionProgress->setRange(0, 1);
381   ui.sessionProgress->setValue(0);
382   ui.networksProgress->setRange(0, 1);
383   ui.networksProgress->setValue(0);
384   ui.channelsProgress->setRange(0, 1);
385   ui.channelsProgress->setValue(0);
386   ui.ircUsersProgress->setRange(0, 1);
387   ui.ircUsersProgress->setValue(0);
388
389   ui.stackedWidget->setCurrentWidget(ui.syncPage);
390   // clean up old page
391   ui.loginGroup->setTitle(tr("Login"));
392   ui.user->setEnabled(true);
393   ui.password->setEnabled(true);
394   ui.rememberPasswd->setEnabled(true);
395   ui.loginButtonBox->button(QDialogButtonBox::Ok)->setEnabled(true);
396 }
397
398 void CoreConnectDlg::coreSessionProgress(quint32 val, quint32 max) {
399   ui.sessionProgress->setRange(0, max);
400   ui.sessionProgress->setValue(val);
401
402 }
403
404 void CoreConnectDlg::coreNetworksProgress(quint32 val, quint32 max) {
405   if(max == 0) {
406     ui.networksProgress->setFormat("0/0");
407     ui.networksProgress->setRange(0, 1);
408     ui.networksProgress->setValue(1);
409   } else {
410     ui.networksProgress->setFormat("%v/%m");
411     ui.networksProgress->setRange(0, max);
412     ui.networksProgress->setValue(val);
413   }
414 }
415
416 void CoreConnectDlg::coreChannelsProgress(quint32 val, quint32 max) {
417   if(max == 0) {
418     ui.channelsProgress->setFormat("0/0");
419     ui.channelsProgress->setRange(0, 1);
420     ui.channelsProgress->setValue(1);
421   } else {
422     ui.channelsProgress->setFormat("%v/%m");
423     ui.channelsProgress->setRange(0, max);
424     ui.channelsProgress->setValue(val);
425   }
426 }
427
428 void CoreConnectDlg::coreIrcUsersProgress(quint32 val, quint32 max) {
429   if(max == 0) {
430     ui.ircUsersProgress->setFormat("0/0");
431     ui.ircUsersProgress->setRange(0, 1);
432     ui.ircUsersProgress->setValue(1);
433   } else {
434     if(val % 100) return;
435     ui.ircUsersProgress->setFormat("%v/%m");
436     ui.ircUsersProgress->setRange(0, max);
437     ui.ircUsersProgress->setValue(val);
438   }
439 }
440
441 void CoreConnectDlg::syncFinished() {
442   if(!wizard) accept();
443   else {
444     hide();
445     disconnect(wizard, 0, this, 0);
446     connect(wizard, SIGNAL(finished(int)), this, SLOT(accept()));
447   }
448 }
449
450 /*****************************************************************************************
451  * CoreAccountEditDlg
452  *****************************************************************************************/
453
454 CoreAccountEditDlg::CoreAccountEditDlg(AccountId id, const QVariantMap &acct, const QStringList &_existing, QWidget *parent) : QDialog(parent) {
455   ui.setupUi(this);
456   existing = _existing;
457   account = acct;
458   if(id.isValid()) {
459     existing.removeAll(acct["AccountName"].toString());
460     ui.host->setText(acct["Host"].toString());
461     ui.port->setValue(acct["Port"].toUInt());
462     ui.useInternal->setChecked(acct["UseInternal"].toBool());
463     ui.accountName->setText(acct["AccountName"].toString());
464   } else {
465     setWindowTitle(tr("Add Core Account"));
466   }
467 }
468
469 QVariantMap CoreAccountEditDlg::accountData() {
470   account["AccountName"] = ui.accountName->text();
471   account["Host"] = ui.host->text();
472   account["Port"] = ui.port->value();
473   account["UseInternal"] = ui.useInternal->isChecked();
474   return account;
475 }
476
477 void CoreAccountEditDlg::setWidgetStates() {
478   bool ok = !ui.accountName->text().isEmpty() && !existing.contains(ui.accountName->text()) && (ui.useInternal->isChecked() || !ui.host->text().isEmpty());
479   ui.buttonBox->button(QDialogButtonBox::Ok)->setEnabled(ok);
480 }
481
482 void CoreAccountEditDlg::on_host_textChanged(const QString &text) {
483   Q_UNUSED(text);
484   setWidgetStates();
485 }
486
487 void CoreAccountEditDlg::on_accountName_textChanged(const QString &text) {
488   Q_UNUSED(text);
489   setWidgetStates();
490 }
491
492 void CoreAccountEditDlg::on_useRemote_toggled(bool state) {
493   Q_UNUSED(state);
494   setWidgetStates();
495 }