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/logdlg.cpp

621 lines
20 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 "logdlg.h"
#include <tqcombobox.h>
#include <tqfile.h>
#include <tqfileinfo.h>
#include <tqlabel.h>
#include <tqlayout.h>
#include <tqpushbutton.h>
#include <tqtabwidget.h>
#include <tqtextedit.h>
#include <tqwhatsthis.h>
#include <tdeconfig.h>
#include <kdebug.h>
#include <tdefiledialog.h>
#include <kfinddialog.h>
#include <tdeglobalsettings.h>
#include <kiconloader.h>
#include <tdelistviewsearchline.h>
#include <tdelocale.h>
#include <tdemessagebox.h>
#include <kprocess.h>
#include <krfcdate.h>
#include <krun.h>
#include <kurl.h>
#include "cvsservice_stub.h"
#include "annotatedlg.h"
#include "annotatectl.h"
#include "diffdlg.h"
#include "loginfo.h"
#include "loglist.h"
#include "logplainview.h"
#include "logtree.h"
#include "misc.h"
#include "progressdlg.h"
#include "patchoptiondlg.h"
LogDialog::LogDialog(TDEConfig& cfg, TQWidget *parent, const char *name)
: KDialogBase(parent, name, false, TQString(),
Ok | Apply | Close | Help | User1 | User2 | User3, Close, true,
KGuiItem(i18n("&Annotate")),
KGuiItem(i18n("&Diff"), "vcs_diff"),
KGuiItem(i18n("&Find..."), "edit-find"))
, cvsService(0)
, partConfig(cfg)
{
TQSplitter *splitter = new TQSplitter(Qt::Vertical, this);
setMainWidget(splitter);
tree = new LogTreeView(this);
connect( tree, TQT_SIGNAL(revisionClicked(TQString,bool)),
this, TQT_SLOT(revisionSelected(TQString,bool)) );
TQWidget* listWidget = new TQWidget(this);
TQVBoxLayout* listLayout = new TQVBoxLayout(listWidget);
TQHBoxLayout* searchLayout = new TQHBoxLayout(listLayout);
searchLayout->setMargin(KDialog::spacingHint());
searchLayout->setSpacing(KDialog::spacingHint());
list = new LogListView(partConfig, listWidget);
listLayout->addWidget(list, 1);
TDEListViewSearchLine* searchLine = new TDEListViewSearchLine(listWidget, list);
TQLabel* searchLabel = new TQLabel(searchLine, i18n("S&earch:"), listWidget);
searchLayout->addWidget(searchLabel);
searchLayout->addWidget(searchLine, 1);
connect( list, TQT_SIGNAL(revisionClicked(TQString,bool)),
this, TQT_SLOT(revisionSelected(TQString,bool)) );
plain = new LogPlainView(this);
connect( plain, TQT_SIGNAL(revisionClicked(TQString,bool)),
this, TQT_SLOT(revisionSelected(TQString,bool)) );
tabWidget = new TQTabWidget(splitter);
tabWidget->addTab(tree, i18n("&Tree"));
tabWidget->addTab(listWidget, i18n("&List"));
tabWidget->addTab(plain, i18n("CVS &Output"));
connect(tabWidget, TQT_SIGNAL(currentChanged(TQWidget*)),
this, TQT_SLOT(tabChanged(TQWidget*)));
TQWhatsThis::add(tree, i18n("Choose revision A by clicking with the left "
"mouse button,\nrevision B by clicking with "
"the middle mouse button."));
items.setAutoDelete(true);
tags.setAutoDelete(true);
TQWidget *mainWidget = new TQWidget(splitter);
TQBoxLayout *layout = new TQVBoxLayout(mainWidget, 0, spacingHint());
for (int i = 0; i < 2; ++i)
{
if ( i == 1 )
{
TQFrame *frame = new TQFrame(mainWidget);
frame->setFrameStyle(TQFrame::HLine | TQFrame::Sunken);
layout->addWidget(frame);
}
TQGridLayout *grid = new TQGridLayout(layout);
grid->setRowStretch(0, 0);
grid->setRowStretch(1, 0);
grid->setRowStretch(2, 1);
grid->setColStretch(0, 0);
grid->setColStretch(1, 1);
grid->setColStretch(2, 0);
grid->setColStretch(3, 1);
grid->setColStretch(4, 2);
TQString versionident = (i==0)? i18n("Revision A:") : i18n("Revision B:");
TQLabel *versionlabel = new TQLabel(versionident, mainWidget);
grid->addWidget(versionlabel, 0, 0);
revbox[i] = new TQLabel(mainWidget);
revbox[i]->setFrameStyle(TQFrame::Panel | TQFrame::Sunken);
grid->addWidget(revbox[i], 0, 1, TQt::AlignVCenter);
TQLabel *selectlabel = new TQLabel(i18n("Select by tag:"), mainWidget);
grid->addWidget(selectlabel, 0, 2);
tagcombo[i] = new TQComboBox(mainWidget);
TQFontMetrics fm(tagcombo[i]->fontMetrics());
tagcombo[i]->setMinimumWidth(fm.width("X")*20);
grid->addWidget(tagcombo[i], 0, 3);
TQLabel *authorlabel = new TQLabel(i18n("Author:"), mainWidget);
grid->addWidget(authorlabel, 1, 0);
authorbox[i] = new TQLabel(mainWidget);
authorbox[i]->setFrameStyle(TQFrame::Panel | TQFrame::Sunken);
grid->addWidget(authorbox[i], 1, 1);
TQLabel *datelabel = new TQLabel(i18n("Date:"), mainWidget);
grid->addWidget(datelabel, 1, 2);
datebox[i] = new TQLabel(mainWidget);
datebox[i]->setFrameStyle(TQFrame::Panel | TQFrame::Sunken);
grid->addWidget(datebox[i], 1, 3);
TQLabel *commentlabel = new TQLabel(i18n("Comment/Tags:"), mainWidget);
grid->addWidget(commentlabel, 2, 0);
commentbox[i] = new TQTextEdit(mainWidget);
commentbox[i]->setReadOnly(true);
commentbox[i]->setTextFormat(TQt::PlainText);
fm = commentbox[i]->fontMetrics();
commentbox[i]->setMinimumHeight(2*fm.lineSpacing()+10);
grid->addMultiCellWidget(commentbox[i], 2, 2, 1, 3);
tagsbox[i] = new TQTextEdit(mainWidget);
tagsbox[i]->setReadOnly(true);
tagsbox[i]->setMinimumHeight(2*fm.lineSpacing()+10);
grid->addWidget(tagsbox[i], 2, 4);
}
TQWhatsThis::add(revbox[0], i18n("This revision is used when you click "
"Annotate.\nIt is also used as the first "
"item of a Diff operation."));
TQWhatsThis::add(revbox[1], i18n("This revision is used as the second "
"item of a Diff operation."));
connect( tagcombo[0], TQT_SIGNAL(activated(int)),
this, TQT_SLOT(tagASelected(int)) );
connect( tagcombo[1], TQT_SIGNAL(activated(int)),
this, TQT_SLOT(tagBSelected(int)) );
connect( this, TQT_SIGNAL(user1Clicked()),
this, TQT_SLOT(annotateClicked()) );
connect( this, TQT_SIGNAL(user2Clicked()),
this, TQT_SLOT(diffClicked()) );
connect( this, TQT_SIGNAL(user3Clicked()),
this, TQT_SLOT(findClicked()) );
setButtonGuiItem(Ok, KGuiItem(i18n("to view something", "&View"),"document-open"));
setButtonGuiItem(Apply, KGuiItem(i18n("Create Patch...")));
setHelp("browsinglogs");
setWFlags(TQt::WDestructiveClose | getWFlags());
TQSize size = configDialogSize(partConfig, "LogDialog");
resize(size);
TDEConfigGroupSaver cs(&partConfig, "LogDialog");
tabWidget->setCurrentPage(partConfig.readNumEntry("ShowTab", 0));
updateButtons();
}
LogDialog::~LogDialog()
{
saveDialogSize(partConfig, "LogDialog");
TDEConfigGroupSaver cs(&partConfig, "LogDialog");
partConfig.writeEntry("ShowTab", tabWidget->currentPageIndex());
}
bool LogDialog::parseCvsLog(CvsService_stub* service, const TQString& fileName)
{
TQString rev;
Cervisia::LogInfo logInfo;
enum { Begin, Tags, Admin, Revision,
Author, Branches, Comment, Finished } state;
// remember DCOP reference and file name for diff or annotate
cvsService = service;
filename = fileName;
setCaption(i18n("CVS Log: %1").arg(filename));
DCOPRef job = cvsService->log(filename);
if( !cvsService->ok() )
return false;
ProgressDialog dlg(this, "Logging", job, "log", i18n("CVS Log"));
if( !dlg.execute() )
return false;
// process cvs log output
state = Begin;
TQString line;
while( dlg.getLine(line) )
{
switch( state )
{
case Begin:
if( line == "symbolic names:" )
state = Tags;
break;
case Tags:
if( line[0] == '\t' )
{
const TQStringList strlist(splitLine(line, ':'));
rev = strlist[1].simplifyWhiteSpace();
const TQString tag(strlist[0].simplifyWhiteSpace());
TQString branchpoint;
int pos1, pos2;
if( (pos2 = rev.findRev('.')) > 0 &&
(pos1 = rev.findRev('.', pos2-1)) > 0 &&
rev.mid(pos1+1, pos2-pos1-1) == "0" )
{
// For a branch tag 2.10.0.6, we want:
// branchpoint = "2.10"
// rev = "2.10.6"
branchpoint = rev.left(pos1);
rev.remove(pos1+1, pos2-pos1);
}
if( rev != "1.1.1" )
{
LogDialogTagInfo *taginfo = new LogDialogTagInfo;
taginfo->rev = rev;
taginfo->tag = tag;
taginfo->branchpoint = branchpoint;
tags.append(taginfo);
}
}
else
{
state = Admin;
}
break;
case Admin:
if( line == "----------------------------" )
{
state = Revision;
}
break;
case Revision:
logInfo.m_revision = rev = line.section(' ', 1, 1);
state = Author;
break;
case Author:
{
TQStringList strList = TQStringList::split(";", line);
// convert date into ISO format (YYYY-MM-DDTHH:MM:SS)
int len = strList[0].length();
TQString dateTimeStr = strList[0].right(len-6); // remove 'date: '
dateTimeStr.replace('/', '-');
TQString date = dateTimeStr.section(' ', 0, 0);
TQString time = dateTimeStr.section(' ', 1, 1);
logInfo.m_dateTime.setTime_t(KRFCDate::parseDateISO8601(date + 'T' + time));
logInfo.m_author = strList[1].section(':', 1, 1).stripWhiteSpace();
state = Branches;
}
break;
case Branches:
if( !line.startsWith("branches:") )
{
logInfo.m_comment = line;
state = Comment;
}
break;
case Comment:
if( line == "----------------------------" )
{
state = Revision;
}
else if( line == "=============================================================================" )
{
state = Finished;
}
if( state == Comment ) // still in message
logInfo.m_comment += '\n' + line;
else
{
// Create tagcomment
TQString branchrev;
int pos1, pos2;
// 1.60.x.y => revision belongs to branch 1.60.0.x
if( (pos2 = rev.findRev('.')) > 0 &&
(pos1 = rev.findRev('.', pos2-1)) > 0 )
branchrev = rev.left(pos2);
// Build Cervisia::TagInfo for logInfo
TQPtrListIterator<LogDialogTagInfo> it(tags);
for( ; it.current(); ++it )
{
if( rev == it.current()->rev )
{
// This never matches branch tags...
logInfo.m_tags.push_back(Cervisia::TagInfo(it.current()->tag,
Cervisia::TagInfo::Tag));
}
if( rev == it.current()->branchpoint )
{
logInfo.m_tags.push_back(Cervisia::TagInfo(it.current()->tag,
Cervisia::TagInfo::Branch));
}
if( branchrev == it.current()->rev )
{
// ... and this never matches ordinary tags :-)
logInfo.m_tags.push_back(Cervisia::TagInfo(it.current()->tag,
Cervisia::TagInfo::OnBranch));
}
}
plain->addRevision(logInfo);
tree->addRevision(logInfo);
list->addRevision(logInfo);
items.append(new Cervisia::LogInfo(logInfo));
// reset for next entry
logInfo = Cervisia::LogInfo();
}
break;
case Finished:
;
}
}
tagcombo[0]->insertItem(TQString());
tagcombo[1]->insertItem(TQString());
TQPtrListIterator<LogDialogTagInfo> it(tags);
for( ; it.current(); ++it )
{
TQString str = it.current()->tag;
if( !it.current()->branchpoint.isEmpty() )
str += i18n(" (Branchpoint)");
tagcombo[0]->insertItem(str);
tagcombo[1]->insertItem(str);
}
plain->scrollToTop();
tree->collectConnections();
tree->recomputeCellSizes();
return true; // successful
}
void LogDialog::slotOk()
{
// make sure that the user selected a revision
if( selectionA.isEmpty() && selectionB.isEmpty() )
{
KMessageBox::information(this,
i18n("Please select revision A or B first."), "Cervisia");
return;
}
// retrieve the selected revision
TQString revision;
if( !selectionA.isEmpty() )
revision = selectionA;
else
revision = selectionB;
// create a temporary file
const TQString suffix("-" + revision + "-" + TQFileInfo(filename).fileName());
const TQString tempFileName(::tempFileName(suffix));
// retrieve the file with the selected revision from cvs
// and save the content into the temporary file
DCOPRef job = cvsService->downloadRevision(filename, revision, tempFileName);
if( !cvsService->ok() )
return;
ProgressDialog dlg(this, "View", job, "view", i18n("View File"));
if( dlg.execute() )
{
// make file read-only
chmod(TQFile::encodeName(tempFileName), 0400);
// open file in preferred editor
KURL url;
url.setPath(tempFileName);
(void) new KRun(url, 0, true, false);
}
}
void LogDialog::slotApply()
{
if( selectionA.isEmpty() )
{
KMessageBox::information(this,
i18n("Please select revision A or revisions A and B first."),
"Cervisia");
return;
}
Cervisia::PatchOptionDialog optionDlg;
if( optionDlg.exec() == KDialogBase::Rejected )
return;
TQString format = optionDlg.formatOption();
TQString diffOptions = optionDlg.diffOptions();
DCOPRef job = cvsService->diff(filename, selectionA, selectionB, diffOptions,
format);
if( !cvsService->ok() )
return;
ProgressDialog dlg(this, "Diff", job, "", i18n("CVS Diff"));
if( !dlg.execute() )
return;
TQString fileName = KFileDialog::getSaveFileName();
if( fileName.isEmpty() )
return;
if( !Cervisia::CheckOverwrite(fileName) )
return;
TQFile f(fileName);
if( !f.open(IO_WriteOnly) )
{
KMessageBox::sorry(this,
i18n("Could not open file for writing."),
"Cervisia");
return;
}
TQTextStream t(&f);
TQString line;
while( dlg.getLine(line) )
t << line << '\n';
f.close();
}
void LogDialog::findClicked()
{
KFindDialog dlg(this);
if( dlg.exec() == KDialogBase::Accepted )
plain->searchText(dlg.options(), dlg.pattern());
}
void LogDialog::diffClicked()
{
if (selectionA.isEmpty())
{
KMessageBox::information(this,
i18n("Please select revision A or revisions A and B first."),
"Cervisia");
return;
}
// Non-modal dialog
DiffDialog *l = new DiffDialog(partConfig);
if (l->parseCvsDiff(cvsService, filename, selectionA, selectionB))
l->show();
else
delete l;
}
void LogDialog::annotateClicked()
{
AnnotateDialog *l = new AnnotateDialog(partConfig);
AnnotateController ctl(l, cvsService);
ctl.showDialog(filename, selectionA);
}
void LogDialog::revisionSelected(TQString rev, bool rmb)
{
TQPtrListIterator<Cervisia::LogInfo> it(items);
for (; it.current(); ++it)
if (it.current()->m_revision == rev)
{
if (rmb)
selectionB = rev;
else
selectionA = rev;
revbox[rmb?1:0]->setText(rev);
authorbox[rmb?1:0]->setText(it.current()->m_author);
datebox[rmb?1:0]->setText(it.current()->dateTimeToString());
commentbox[rmb?1:0]->setText(it.current()->m_comment);
tagsbox[rmb?1:0]->setText(it.current()->tagsToString());
tree->setSelectedPair(selectionA, selectionB);
list->setSelectedPair(selectionA, selectionB);
updateButtons();
return;
}
kdDebug(8050) << "Internal error: Revision not found " << rev << "." << endl;
}
void LogDialog::tagSelected(LogDialogTagInfo* tag, bool rmb)
{
if (tag->branchpoint.isEmpty())
revisionSelected(tag->rev, rmb);
else
revisionSelected(tag->branchpoint, rmb);
}
void LogDialog::updateButtons()
{
// no versions selected?
if( selectionA.isEmpty() && selectionB.isEmpty() )
{
enableButton(User1, true); // annotate
enableButton(User2, false); // diff
enableButtonOK(false); // view
enableButtonApply(false); // create patch
}
// both versions selected?
else if( !selectionA.isEmpty() && !selectionB.isEmpty() )
{
enableButton(User1, false); // annotate
enableButton(User2, true); // diff
enableButtonOK(false); // view
enableButtonApply(true); // create patch
}
// only single version selected?
else
{
enableButton(User1, true); // annotate
enableButton(User2, true); // diff
enableButtonOK(true); // view
enableButtonApply(true); // create patch
}
}
void LogDialog::tagASelected(int n)
{
if (n)
tagSelected(tags.at(n-1), false);
}
void LogDialog::tagBSelected(int n)
{
if (n)
tagSelected(tags.at(n-1), true);
}
void LogDialog::tabChanged(TQWidget* w)
{
bool isPlainView = (w == plain);
showButton(User3, isPlainView);
}
#include "logdlg.moc"
// Local Variables:
// c-basic-offset: 4
// End: