You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
277 lines
7.0 KiB
277 lines
7.0 KiB
/*
|
|
* Copyright (c) 1999-2002 Bernd Gehrmann <bernd@mail.berlios.de>
|
|
* Copyright (c) 2002-2004 Christian Loose <christian.loose@kdemail.net>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include "progressdlg.h"
|
|
|
|
#include <qlabel.h>
|
|
#include <qlayout.h>
|
|
#include <qstring.h>
|
|
#include <qstringlist.h>
|
|
#include <qtimer.h>
|
|
#include <qvbox.h>
|
|
|
|
#include <cvsjob_stub.h>
|
|
#include <dcopref.h>
|
|
#include <kanimwidget.h>
|
|
#include <kapplication.h>
|
|
#include <kconfig.h>
|
|
|
|
#include "cervisiasettings.h"
|
|
|
|
|
|
struct ProgressDialog::Private
|
|
{
|
|
bool isCancelled;
|
|
bool isShown;
|
|
bool hasError;
|
|
|
|
CvsJob_stub* cvsJob;
|
|
QString buffer;
|
|
QString errorId1, errorId2;
|
|
QStringList output;
|
|
|
|
QTimer* timer;
|
|
KAnimWidget* gear;
|
|
QListBox* resultbox;
|
|
};
|
|
|
|
|
|
ProgressDialog::ProgressDialog(QWidget* parent, const QString& heading,
|
|
const DCOPRef& job, const QString& errorIndicator,
|
|
const QString& caption)
|
|
: KDialogBase(parent, 0, true, caption, Cancel, Cancel, true)
|
|
, DCOPObject()
|
|
, d(new Private)
|
|
{
|
|
// initialize private data
|
|
d->isCancelled = false;
|
|
d->isShown = false;
|
|
d->hasError = false;
|
|
|
|
d->cvsJob = new CvsJob_stub(job);
|
|
d->buffer = "";
|
|
|
|
d->errorId1 = "cvs " + errorIndicator + ":";
|
|
d->errorId2 = "cvs [" + errorIndicator + " aborted]:";
|
|
|
|
setupGui(heading);
|
|
}
|
|
|
|
|
|
ProgressDialog::~ProgressDialog()
|
|
{
|
|
delete d->cvsJob;
|
|
delete d;
|
|
}
|
|
|
|
|
|
void ProgressDialog::setupGui(const QString& heading)
|
|
{
|
|
QVBox* vbox = makeVBoxMainWidget();
|
|
vbox->setSpacing(10);
|
|
|
|
QWidget* headingBox = new QWidget(vbox);
|
|
QHBoxLayout* hboxLayout = new QHBoxLayout(headingBox);
|
|
|
|
QLabel* textLabel = new QLabel(heading, headingBox);
|
|
textLabel->setMinimumWidth(textLabel->sizeHint().width());
|
|
textLabel->setFixedHeight(textLabel->sizeHint().height());
|
|
hboxLayout->addWidget(textLabel);
|
|
hboxLayout->addStretch();
|
|
|
|
d->gear = new KAnimWidget(QString("kde"), 32, headingBox);
|
|
d->gear->setFixedSize(32, 32);
|
|
hboxLayout->addWidget(d->gear);
|
|
|
|
d->resultbox = new QListBox(vbox);
|
|
d->resultbox->setSelectionMode(QListBox::NoSelection);
|
|
QFontMetrics fm(d->resultbox->fontMetrics());
|
|
d->resultbox->setMinimumSize(fm.width("0")*70, fm.lineSpacing()*8);
|
|
|
|
resize(sizeHint());
|
|
}
|
|
|
|
|
|
bool ProgressDialog::execute()
|
|
{
|
|
// get command line and display it
|
|
QString cmdLine = d->cvsJob->cvsCommand();
|
|
d->resultbox->insertItem(cmdLine);
|
|
|
|
// establish connections to the signals of the cvs job
|
|
connectDCOPSignal(d->cvsJob->app(), d->cvsJob->obj(), "jobExited(bool, int)",
|
|
"slotJobExited(bool, int)", true);
|
|
connectDCOPSignal(d->cvsJob->app(), d->cvsJob->obj(), "receivedStdout(QString)",
|
|
"slotReceivedOutputNonGui(QString)", true);
|
|
connectDCOPSignal(d->cvsJob->app(), d->cvsJob->obj(), "receivedStderr(QString)",
|
|
"slotReceivedOutputNonGui(QString)", true);
|
|
|
|
// we wait for 4 seconds (or the timeout set by the user) before we
|
|
// force the dialog to show up
|
|
d->timer = new QTimer(this);
|
|
connect(d->timer, SIGNAL(timeout()), this, SLOT(slotTimeoutOccurred()));
|
|
d->timer->start(CervisiaSettings::timeout(), true);
|
|
|
|
bool started = d->cvsJob->execute();
|
|
if( !started )
|
|
return false;
|
|
|
|
QApplication::setOverrideCursor(waitCursor);
|
|
kapp->enter_loop();
|
|
if (QApplication::overrideCursor())
|
|
QApplication::restoreOverrideCursor();
|
|
|
|
return !d->isCancelled;
|
|
}
|
|
|
|
|
|
bool ProgressDialog::getLine(QString& line)
|
|
{
|
|
if( d->output.isEmpty() )
|
|
return false;
|
|
|
|
line = d->output.first();
|
|
d->output.remove(d->output.begin());
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
QStringList ProgressDialog::getOutput() const
|
|
{
|
|
return d->output;
|
|
}
|
|
|
|
|
|
void ProgressDialog::slotReceivedOutputNonGui(QString buffer)
|
|
{
|
|
d->buffer += buffer;
|
|
|
|
processOutput();
|
|
if( d->hasError )
|
|
{
|
|
stopNonGuiPart();
|
|
startGuiPart();
|
|
}
|
|
}
|
|
|
|
|
|
void ProgressDialog::slotReceivedOutput(QString buffer)
|
|
{
|
|
d->buffer += buffer;
|
|
processOutput();
|
|
}
|
|
|
|
|
|
void ProgressDialog::slotJobExited(bool normalExit, int status)
|
|
{
|
|
Q_UNUSED(status)
|
|
|
|
if( !d->isShown )
|
|
stopNonGuiPart();
|
|
|
|
d->gear->stop();
|
|
if( !d->buffer.isEmpty() )
|
|
{
|
|
d->buffer += '\n';
|
|
processOutput();
|
|
}
|
|
|
|
// Close the dialog automatically if there are no
|
|
// error messages or the process has been aborted
|
|
// 'by hand' (e.g. by clicking the cancel button)
|
|
if( !d->hasError || !normalExit )
|
|
kapp->exit_loop();
|
|
}
|
|
|
|
|
|
void ProgressDialog::slotCancel()
|
|
{
|
|
d->isCancelled = true;
|
|
|
|
bool isRunning = d->cvsJob->isRunning();
|
|
if( isRunning )
|
|
d->cvsJob->cancel();
|
|
else
|
|
kapp->exit_loop();
|
|
}
|
|
|
|
|
|
void ProgressDialog::slotTimeoutOccurred()
|
|
{
|
|
stopNonGuiPart();
|
|
startGuiPart();
|
|
}
|
|
|
|
|
|
void ProgressDialog::stopNonGuiPart()
|
|
{
|
|
d->timer->stop();
|
|
|
|
disconnectDCOPSignal(d->cvsJob->app(), d->cvsJob->obj(), "receivedStdout(QString)",
|
|
"slotReceivedOutputNonGui(QString)");
|
|
disconnectDCOPSignal(d->cvsJob->app(), d->cvsJob->obj(), "receivedStderr(QString)",
|
|
"slotReceivedOutputNonGui(QString)");
|
|
|
|
kapp->exit_loop();
|
|
}
|
|
|
|
|
|
void ProgressDialog::startGuiPart()
|
|
{
|
|
connectDCOPSignal(d->cvsJob->app(), d->cvsJob->obj(), "receivedStdout(QString)",
|
|
"slotReceivedOutput(QString)", true);
|
|
connectDCOPSignal(d->cvsJob->app(), d->cvsJob->obj(), "receivedStderr(QString)",
|
|
"slotReceivedOutput(QString)", true);
|
|
|
|
show();
|
|
d->isShown = true;
|
|
|
|
d->gear->start();
|
|
QApplication::restoreOverrideCursor();
|
|
kapp->enter_loop();
|
|
}
|
|
|
|
|
|
void ProgressDialog::processOutput()
|
|
{
|
|
int pos;
|
|
while( (pos = d->buffer.find('\n')) != -1 )
|
|
{
|
|
QString item = d->buffer.left(pos);
|
|
if( item.startsWith(d->errorId1) ||
|
|
item.startsWith(d->errorId2) ||
|
|
item.startsWith("cvs [server aborted]:") )
|
|
{
|
|
d->hasError = true;
|
|
d->resultbox->insertItem(item);
|
|
}
|
|
else if( item.startsWith("cvs server:") )
|
|
d->resultbox->insertItem(item);
|
|
else
|
|
d->output.append(item);
|
|
|
|
// remove item from buffer
|
|
d->buffer.remove(0, pos+1);
|
|
}
|
|
}
|
|
|
|
|
|
#include "progressdlg.moc"
|