/************************************************************************** ** ** This file is part of Qt Creator ** ** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies). ** ** Contact: Nokia Corporation (info@qt.nokia.com) ** ** ** GNU Lesser General Public License Usage ** ** This file may be used under the terms of the GNU Lesser General Public ** License version 2.1 as published by the Free Software Foundation and ** appearing in the file LICENSE.LGPL included in the packaging of this file. ** Please review the following information to ensure the GNU Lesser General ** Public License version 2.1 requirements will be met: ** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Nokia gives you certain additional ** rights. These rights are described in the Nokia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** Other Usage ** ** Alternatively, this file may be used in accordance with the terms and ** conditions contained in a signed written agreement between you and Nokia. ** ** If you have questions regarding the use of this file, please contact ** Nokia at info@qt.nokia.com. ** **************************************************************************/ #include "giteditor.h" #include "annotationhighlighter.h" #include "gitconstants.h" #include "gitplugin.h" #include "gitclient.h" #include "gitsettings.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define CHANGE_PATTERN_8C "[a-f0-9]{7,8}" #define CHANGE_PATTERN_40C "[a-f0-9]{40,40}" namespace Git { namespace Internal { // ------------ GitEditor GitEditor::GitEditor(const VCSBase::VCSBaseEditorParameters *type, QWidget *parent) : VCSBase::VCSBaseEditorWidget(type, parent), m_changeNumberPattern8(QLatin1String(CHANGE_PATTERN_8C)), m_changeNumberPattern40(QLatin1String(CHANGE_PATTERN_40C)) { QTC_ASSERT(m_changeNumberPattern8.isValid(), return); QTC_ASSERT(m_changeNumberPattern40.isValid(), return); setAnnotateRevisionTextFormat(tr("Blame %1")); setAnnotatePreviousRevisionTextFormat(tr("Blame parent revision %1")); if (Git::Constants::debug) qDebug() << "GitEditor::GitEditor" << type->type << type->id; } QSet GitEditor::annotationChanges() const { QSet changes; const QString txt = toPlainText(); if (txt.isEmpty()) return changes; // Hunt for first change number in annotation: ":" QRegExp r(QLatin1String("^("CHANGE_PATTERN_8C") ")); QTC_ASSERT(r.isValid(), return changes); if (r.indexIn(txt) != -1) { changes.insert(r.cap(1)); r.setPattern(QLatin1String("\n("CHANGE_PATTERN_8C") ")); QTC_ASSERT(r.isValid(), return changes); int pos = 0; while ((pos = r.indexIn(txt, pos)) != -1) { pos += r.matchedLength(); changes.insert(r.cap(1)); } } if (Git::Constants::debug) qDebug() << "GitEditor::annotationChanges() returns #" << changes.size(); return changes; } QString GitEditor::changeUnderCursor(const QTextCursor &c) const { QTextCursor cursor = c; // Any number is regarded as change number. cursor.select(QTextCursor::WordUnderCursor); if (!cursor.hasSelection()) return QString(); const QString change = cursor.selectedText(); if (Git::Constants::debug > 1) qDebug() << "GitEditor:::changeUnderCursor:" << change; if (m_changeNumberPattern8.exactMatch(change)) return change; if (m_changeNumberPattern40.exactMatch(change)) return change; return QString(); } VCSBase::DiffHighlighter *GitEditor::createDiffHighlighter() const { const QRegExp filePattern(QLatin1String("^(diff --git a/|index |[+-][+-][+-] [ab]).*$")); return new VCSBase::DiffHighlighter(filePattern); } VCSBase::BaseAnnotationHighlighter *GitEditor::createAnnotationHighlighter(const QSet &changes) const { return new GitAnnotationHighlighter(changes); } QString GitEditor::fileNameFromDiffSpecification(const QTextBlock &inBlock) const { // Check for "+++ b/src/plugins/git/giteditor.cpp" (blame and diff) // Go back chunks. const QString newFileIndicator = QLatin1String("+++ b/"); for (QTextBlock block = inBlock; block.isValid(); block = block.previous()) { QString diffFileName = block.text(); if (diffFileName.startsWith(newFileIndicator)) { diffFileName.remove(0, newFileIndicator.size()); return findDiffFile(diffFileName, GitPlugin::instance()->versionControl()); } } return QString(); } /* Remove the date specification from annotation, which is tabular: \code 8ca887aa (author YYYY-MM-DD HH:MM:SS ) \endcode */ static QByteArray removeAnnotationDate(const QByteArray &b) { if (b.isEmpty()) return QByteArray(); const int parenPos = b.indexOf(')'); if (parenPos == -1) return QByteArray(b); int datePos = parenPos; // Go back from paren for 5 spaces: That is where the date starts. int spaceCount = 0; for (int i = parenPos; i >= 0; --i) { if (b.at(i) == ' ') ++spaceCount; if (spaceCount == 5) { datePos = i + 1; break; } } if (datePos == 0) return QByteArray(b); // Copy over the parts that have not changed into a new byte array Q_ASSERT(b.size() >= parenPos); QByteArray result(b.constData(), datePos); int prevPos = 0; int pos = parenPos; forever { Q_ASSERT(prevPos < pos); if (prevPos != 0) result.append(b.constData() + prevPos, pos - prevPos); prevPos = pos; Q_ASSERT(prevPos != 0); if (pos == b.size()) break; pos = b.indexOf('\n', pos + 1); if (pos == -1) pos = b.size(); } return result; } void GitEditor::setPlainTextDataFiltered(const QByteArray &a) { QByteArray array = a; // If desired, filter out the date from annotation const bool omitAnnotationDate = contentType() == VCSBase::AnnotateOutput && GitPlugin::instance()->settings().omitAnnotationDate; if (omitAnnotationDate) array = removeAnnotationDate(a); setPlainTextData(array); } void GitEditor::commandFinishedGotoLine(bool ok, int /* exitCode */, const QVariant &v) { if (ok && v.type() == QVariant::Int) { const int line = v.toInt(); if (line >= 0) gotoLine(line); } } QStringList GitEditor::annotationPreviousVersions(const QString &revision) const { QStringList revisions; QString errorMessage; GitClient *client = GitPlugin::instance()->gitClient(); const QFileInfo fi(source()); const QString workingDirectory = fi.absolutePath(); // Get the SHA1's of the file. if (!client->synchronousParentRevisions(workingDirectory, QStringList(fi.fileName()), revision, &revisions, &errorMessage)) { VCSBase::VCSBaseOutputWindow::instance()->appendSilently(errorMessage); return QStringList(); } // Format verbose, SHA1 being first token QStringList descriptions; if (!client->synchronousShortDescriptions(workingDirectory, revisions, &descriptions, &errorMessage)) { VCSBase::VCSBaseOutputWindow::instance()->appendSilently(errorMessage); return QStringList(); } return descriptions; } } // namespace Internal } // namespace Git