/* * Copyright (C) 1999-2002 Bernd Gehrmann * bernd@mail.berlios.de * Copyright (c) 2002-2004 Christian Loose * * 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #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(TQt::Vertical, this); setMainWidget(splitter); tree = new LogTreeView(this); connect( tree, TQ_SIGNAL(revisionClicked(TQString,bool)), this, TQ_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, TQ_SIGNAL(revisionClicked(TQString,bool)), this, TQ_SLOT(revisionSelected(TQString,bool)) ); plain = new LogPlainView(this); connect( plain, TQ_SIGNAL(revisionClicked(TQString,bool)), this, TQ_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, TQ_SIGNAL(currentChanged(TQWidget*)), this, TQ_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], TQ_SIGNAL(activated(int)), this, TQ_SLOT(tagASelected(int)) ); connect( tagcombo[1], TQ_SIGNAL(activated(int)), this, TQ_SLOT(tagBSelected(int)) ); connect( this, TQ_SIGNAL(user1Clicked()), this, TQ_SLOT(annotateClicked()) ); connect( this, TQ_SIGNAL(user2Clicked()), this, TQ_SLOT(diffClicked()) ); connect( this, TQ_SIGNAL(user3Clicked()), this, TQ_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 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 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 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"