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.
tdesdk/cervisia/diffdlg.cpp

510 lines
14 KiB

/*
* Copyright (C) 1999-2002 Bernd Gehrmann
* bernd@mail.berlios.de
*
* 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 "diffdlg.h"
#include <tqpushbutton.h>
#include <tqcheckbox.h>
#include <tqcombobox.h>
#include <tqlabel.h>
#include <tqlayout.h>
#include <tqkeycode.h>
#include <tqfileinfo.h>
#include <tqregexp.h>
#include <kconfig.h>
#include <kfiledialog.h>
#include <klocale.h>
#include <kmessagebox.h>
#include <ktempfile.h>
#include <kprocess.h>
#include "cvsservice_stub.h"
#include "repository_stub.h"
#include "misc.h"
#include "progressdlg.h"
#include "diffview.h"
DiffDialog::DiffDialog(KConfig& cfg, TQWidget *parent, const char *name, bool modal)
: KDialogBase(parent, name, modal, TQString(),
Close | Help | User1, Close, true, KStdGuiItem::saveAs())
, partConfig(cfg)
{
items.setAutoDelete(true);
markeditem = -1;
TQFrame* mainWidget = makeMainWidget();
TQBoxLayout *tqlayout = new TQVBoxLayout(mainWidget, 0, spacingHint());
TQGridLayout *pairtqlayout = new TQGridLayout(tqlayout);
pairtqlayout->setRowStretch(0, 0);
pairtqlayout->setRowStretch(1, 1);
pairtqlayout->setColStretch(1, 0);
pairtqlayout->addColSpacing(1, 16);
pairtqlayout->setColStretch(0, 10);
pairtqlayout->setColStretch(2, 10);
revlabel1 = new TQLabel(mainWidget);
pairtqlayout->addWidget(revlabel1, 0, 0);
revlabel2 = new TQLabel(mainWidget);
pairtqlayout->addWidget(revlabel2, 0, 2);
diff1 = new DiffView(cfg, true, false, mainWidget);
diff2 = new DiffView(cfg, true, true, mainWidget);
DiffZoomWidget *zoom = new DiffZoomWidget(cfg, mainWidget);
zoom->setDiffView(diff2);
pairtqlayout->addWidget(diff1, 1, 0);
pairtqlayout->addWidget(zoom, 1, 1);
pairtqlayout->addWidget(diff2, 1, 2);
diff1->setPartner(diff2);
diff2->setPartner(diff1);
syncbox = new TQCheckBox(i18n("Synchronize scroll bars"), mainWidget);
syncbox->setChecked(true);
connect( syncbox, TQT_SIGNAL(toggled(bool)),
this, TQT_SLOT(toggleSynchronize(bool)) );
itemscombo = new TQComboBox(mainWidget);
itemscombo->insertItem(TQString());
connect( itemscombo, TQT_SIGNAL(activated(int)),
this, TQT_SLOT(comboActivated(int)) );
nofnlabel = new TQLabel(mainWidget);
// avoids auto resize when the text is changed
nofnlabel->setMinimumWidth(fontMetrics().width(i18n("%1 differences").tqarg(10000)));
backbutton = new TQPushButton(TQString::fromLatin1("&<<"), mainWidget);
connect( backbutton, TQT_SIGNAL(clicked()), TQT_SLOT(backClicked()) );
forwbutton = new TQPushButton(TQString::fromLatin1("&>>"), mainWidget);
connect( forwbutton, TQT_SIGNAL(clicked()), TQT_SLOT(forwClicked()) );
connect( this, TQT_SIGNAL(user1Clicked()), TQT_SLOT(saveAsClicked()) );
TQBoxLayout *buttontqlayout = new TQHBoxLayout(tqlayout);
buttontqlayout->addWidget(syncbox, 0);
buttontqlayout->addStretch(4);
buttontqlayout->addWidget(itemscombo);
buttontqlayout->addStretch(1);
buttontqlayout->addWidget(nofnlabel);
buttontqlayout->addStretch(1);
buttontqlayout->addWidget(backbutton);
buttontqlayout->addWidget(forwbutton);
setHelp("diff");
setWFlags(TQt::WDestructiveClose | getWFlags());
TQSize size = configDialogSize(partConfig, "DiffDialog");
resize(size);
KConfigGroupSaver cs(&partConfig, "DiffDialog");
syncbox->setChecked(partConfig.readBoolEntry("Sync"));
}
DiffDialog::~DiffDialog()
{
saveDialogSize(partConfig, "DiffDialog");
KConfigGroupSaver cs(&partConfig, "DiffDialog");
partConfig.writeEntry("Sync", syncbox->isChecked());
}
void DiffDialog::keyPressEvent(TQKeyEvent *e)
{
switch (e->key())
{
case Key_Up:
diff1->up();
diff2->up();
break;
case Key_Down:
diff1->down();
diff2->down();
break;
case Key_Next:
diff1->next();
diff2->next();
break;
case Key_Prior:
diff1->prior();
diff2->prior();
break;
default:
KDialogBase::keyPressEvent(e);
}
}
void DiffDialog::toggleSynchronize(bool b)
{
diff1->setPartner(b? diff2 : 0);
diff2->setPartner(b? diff1 : 0);
}
void DiffDialog::comboActivated(int index)
{
updateHighlight(index-1);
}
static void interpretRegion(TQString line, int *linenoA, int *linenoB)
{
TQRegExp region( "^@@ -([0-9]+),([0-9]+) \\+([0-9]+),([0-9]+) @@.*$" );
if (!region.exactMatch(line))
return;
*linenoA = region.cap(1).toInt() - 1;
*linenoB = region.cap(3).toInt() - 1;
}
static TQString regionAsString(int linenoA, int linecountA, int linenoB, int linecountB)
{
int lineendA = linenoA+linecountA-1;
int lineendB = linenoB+linecountB-1;
TQString res;
if (linecountB == 0)
res = TQString("%1,%2d%3").tqarg(linenoA).tqarg(lineendA).tqarg(linenoB-1);
else if (linecountA == 0)
res = TQString("%1a%2,%3").tqarg(linenoA-1).tqarg(linenoB).tqarg(lineendB);
else if (linenoA == lineendA)
if (linenoB == lineendB)
res = TQString("%1c%2").tqarg(linenoA).tqarg(linenoB);
else
res = TQString("%1c%2,%3").tqarg(linenoA).tqarg(linenoB).tqarg(lineendB);
else if (linenoB == lineendB)
res = TQString("%1,%2c%3").tqarg(linenoA).tqarg(lineendA).tqarg(linenoB);
else
res = TQString("%1,%2c%3,%4").tqarg(linenoA).tqarg(lineendA).tqarg(linenoB).tqarg(lineendB);
return res;
}
class DiffItem
{
public:
DiffView::DiffType type;
int linenoA, linecountA;
int linenoB, linecountB;
};
bool DiffDialog::parseCvsDiff(CvsService_stub* service, const TQString& fileName,
const TQString &revA, const TQString &revB)
{
TQStringList linesA, linesB;
int linenoA, linenoB;
setCaption(i18n("CVS Diff: %1").tqarg(fileName));
revlabel1->setText( revA.isEmpty()?
i18n("Repository:")
: i18n("Revision ")+revA+":" );
revlabel2->setText( revB.isEmpty()?
i18n("Working dir:")
: i18n("Revision ")+revB+":" );
KConfigGroupSaver cs(&partConfig, "General");
// Ok, this is a hack: When the user wants an external diff
// front end, it is executed from here. Of course, in that
// case this dialog wouldn't have to be created in the first
// place, but this design at least makes the handling trans-
// parent for the calling routines
TQString extdiff = partConfig.readPathEntry("ExternalDiff");
if (!extdiff.isEmpty())
{
callExternalDiff(extdiff, service, fileName, revA, revB);
return false;
}
const TQString diffOptions = partConfig.readEntry("DiffOptions");
const unsigned contextLines = partConfig.readUnsignedNumEntry("ContextLines", 65535);
DCOPRef job = service->diff(fileName, revA, revB, diffOptions, contextLines);
if( !service->ok() )
return false;
ProgressDialog dlg(this, "Diff", job, "diff", i18n("CVS Diff"));
if( !dlg.execute() )
return false;
// remember diff output for "save as" action
m_diffOutput = dlg.getOutput();
TQString line;
while ( dlg.getLine(line) && !line.startsWith("+++"))
;
linenoA = linenoB = 0;
while ( dlg.getLine(line) )
{
// line contains diff region?
if (line.startsWith("@@"))
{
interpretRegion(line, &linenoA, &linenoB);
diff1->addLine(line, DiffView::Separator);
diff2->addLine(line, DiffView::Separator);
continue;
}
if (line.length() < 1)
continue;
TQChar marker = line[0];
line.remove(0, 1);
if (marker == '-')
linesA.append(line);
else if (marker == '+')
linesB.append(line);
else
{
if (!linesA.isEmpty() || !linesB.isEmpty())
{
newDiffHunk(linenoA, linenoB, linesA, linesB);
linesA.clear();
linesB.clear();
}
diff1->addLine(line, DiffView::Unchanged, ++linenoA);
diff2->addLine(line, DiffView::Unchanged, ++linenoB);
}
}
if (!linesA.isEmpty() || !linesB.isEmpty())
newDiffHunk(linenoA, linenoB, linesA, linesB);
// sets the right size as there is no more auto resize in TQComboBox
itemscombo->adjustSize();
updateNofN();
return true;
}
void DiffDialog::newDiffHunk(int& linenoA, int& linenoB,
const TQStringList& linesA, const TQStringList& linesB)
{
DiffItem *item = new DiffItem;
item->linenoA = linenoA+1;
item->linenoB = linenoB+1;
item->linecountA = linesA.count();
item->linecountB = linesB.count();
items.append(item);
const TQString region = regionAsString(linenoA+1, linesA.count(),
linenoB+1, linesB.count());
itemscombo->insertItem(region);
TQStringList::ConstIterator itA = linesA.begin();
TQStringList::ConstIterator itB = linesB.begin();
while (itA != linesA.end() || itB != linesB.end())
{
if (itA != linesA.end())
{
diff1->addLine(*itA, DiffView::Neutral, ++linenoA);
if (itB != linesB.end())
diff2->addLine(*itB, DiffView::Change, ++linenoB);
else
diff2->addLine("", DiffView::Delete);
}
else
{
diff1->addLine("", DiffView::Neutral);
diff2->addLine(*itB, DiffView::Insert, ++linenoB);
}
if (itA != linesA.end())
++itA;
if (itB != linesB.end())
++itB;
}
}
void DiffDialog::callExternalDiff(const TQString& extdiff, CvsService_stub* service,
const TQString& fileName, const TQString &revA,
const TQString &revB)
{
TQString extcmdline = extdiff;
extcmdline += " ";
// create suffix for temporary file (used TQFileInfo to remove path from file name)
const TQString suffix = "-" + TQFileInfo(fileName).fileName();
DCOPRef job;
if (!revA.isEmpty() && !revB.isEmpty())
{
// We're comparing two revisions
TQString revAFilename = tempFileName(suffix+TQString("-")+revA);
TQString revBFilename = tempFileName(suffix+TQString("-")+revB);
// download the files for revision A and B
job = service->downloadRevision(fileName, revA, revAFilename,
revB, revBFilename);
if( !service->ok() )
return;
extcmdline += KProcess::quote(revAFilename);
extcmdline += " ";
extcmdline += KProcess::quote(revBFilename);
}
else
{
// We're comparing to a file, and perhaps one revision
TQString revAFilename = tempFileName(suffix+TQString("-")+revA);
job = service->downloadRevision(fileName, revA, revAFilename);
if( !service->ok() )
return;
extcmdline += KProcess::quote(revAFilename);
extcmdline += " ";
extcmdline += KProcess::quote(TQFileInfo(fileName).absFilePath());
}
ProgressDialog dlg(this, "Diff", job, "diff");
if( dlg.execute() )
{
// call external diff application
// TODO CL maybe use system()?
KProcess proc;
proc.setUseShell(true, "/bin/sh");
proc << extcmdline;
proc.start(KProcess::DontCare);
}
}
void DiffDialog::updateNofN()
{
TQString str;
if (markeditem >= 0)
str = i18n("%1 of %2").tqarg(markeditem+1).tqarg(items.count());
else
str = i18n("%1 differences").tqarg(items.count());
nofnlabel->setText(str);
itemscombo->setCurrentItem(markeditem==-2? 0 : markeditem+1);
backbutton->setEnabled(markeditem != -1);
forwbutton->setEnabled(markeditem != -2 && items.count());
}
void DiffDialog::updateHighlight(int newitem)
{
if (markeditem >= 0)
{
DiffItem *item = items.at(markeditem);
for (int i = item->linenoA; i < item->linenoA+item->linecountA; ++i)
diff1->setInverted(i, false);
for (int i = item->linenoB; i < item->linenoB+item->linecountB; ++i)
diff2->setInverted(i, false);
}
markeditem = newitem;
if (markeditem >= 0)
{
DiffItem *item = items.at(markeditem);
for (int i = item->linenoA; i < item->linenoA+item->linecountA; ++i)
diff1->setInverted(i, true);
for (int i = item->linenoB; i < item->linenoB+item->linecountB; ++i)
diff2->setInverted(i, true);
diff1->setCenterLine(item->linenoA);
diff2->setCenterLine(item->linenoB);
}
diff1->tqrepaint();
diff2->tqrepaint();
updateNofN();
}
void DiffDialog::backClicked()
{
int newitem;
if (markeditem == -1)
return; // internal error (button not disabled)
else if (markeditem == -2) // past end
newitem = items.count()-1;
else
newitem = markeditem-1;
updateHighlight(newitem);
}
void DiffDialog::forwClicked()
{
int newitem;
if (markeditem == -2 || (markeditem == -1 && !items.count()))
return; // internal error (button not disabled)
else if (markeditem+1 == (int)items.count()) // past end
newitem = -2;
else
newitem = markeditem+1;
updateHighlight(newitem);
}
void DiffDialog::saveAsClicked()
{
TQString fileName = KFileDialog::getSaveFileName(TQString(), TQString(), this);
if( fileName.isEmpty() )
return;
if( !Cervisia::CheckOverwrite(fileName, this) )
return;
TQFile f(fileName);
if( !f.open(IO_WriteOnly) )
{
KMessageBox::sorry(this,
i18n("Could not open file for writing."),
"Cervisia");
return;
}
TQTextStream ts(&f);
TQStringList::Iterator it = m_diffOutput.begin();
for( ; it != m_diffOutput.end(); ++it )
ts << *it << "\n";
f.close();
}
#include "diffdlg.moc"
// Local Variables:
// c-basic-offset: 4
// End: