Newer
Older
/**************************************************************************
**
** This file is part of Qt Creator
**
** Copyright (c) 2012 Nokia Corporation and/or its subsidiary(-ies).
** Contact: Nokia Corporation (qt-info@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
**
**************************************************************************/
#include <texteditor/basetexteditor.h>
#include <projectexplorer/project.h>

Kai Koehne
committed
#include <utils/qtcassert.h>
#include <QFileInfo>
#include <QUrl>
#include <QPlainTextEdit>
#include <QTextCursor>
using namespace ProjectExplorer;
using namespace QtSupport;
QtOutputFormatter::QtOutputFormatter(ProjectExplorer::Project *project)
, m_qmlError(QLatin1String("^(?:\\[Qt Message\\] )?" // '[Qt Message] ' prefix (optional, on Symbian)
"(file:///.+" // file url
":\\d+" // colon, line
"(?::\\d+)?)" // colon, column (optional)
":")) // colon
, m_qtAssert(QLatin1String("ASSERT: .* in file (.+, line \\d+)"))
, m_qtAssertX(QLatin1String("ASSERT failure in .*: \".*\", file (.+, line \\d+)"))
, m_qtTestFail(QLatin1String("^ Loc: \\[(.*)\\]"))
if(project) {
m_projectFinder.setProjectFiles(project->files(Project::ExcludeGeneratedFiles));
m_projectFinder.setProjectDirectory(project->projectDirectory());
connect(project, SIGNAL(fileListChanged()),
this, SLOT(updateProjectFileList()));
}
LinkResult QtOutputFormatter::matchLine(const QString &line) const
LinkResult lr;
lr.start = -1;
lr.end = -1;
if (m_qmlError.indexIn(line) != -1) {
lr.href = m_qmlError.cap(1);
lr.start = m_qmlError.pos(1);
lr.end = lr.start + lr.href.length();
} else if (m_qtError.indexIn(line) != -1) {
lr.href = m_qtError.cap(1);
lr.start = m_qtError.pos(1);
lr.end = lr.start + lr.href.length();
} else if (m_qtAssert.indexIn(line) != -1) {
lr.href = m_qtAssert.cap(1);
lr.start = m_qtAssert.pos(1);
lr.end = lr.start + lr.href.length();
} else if (m_qtAssertX.indexIn(line) != -1) {
lr.href = m_qtAssertX.cap(1);
lr.start = m_qtAssertX.pos(1);
lr.end = lr.start + lr.href.length();
} else if (m_qtTestFail.indexIn(line) != -1) {
lr.href = m_qtTestFail.cap(1);
lr.start = m_qtTestFail.pos(1);
lr.end = lr.start + lr.href.length();
void QtOutputFormatter::appendMessage(const QString &txt, Utils::OutputFormat format)
QTextCursor cursor(plainTextEdit()->document());
cursor.movePosition(QTextCursor::End);
cursor.beginEditBlock();
QString deferedText;
int start = 0;
int pos = txt.indexOf(QLatin1Char('\n'));
while (pos != -1) {
// Line identified
if (!m_lastLine.isEmpty()) {
// Line continuation
const QString newPart = txt.mid(start, pos - start + 1);
const QString line = m_lastLine + newPart;
LinkResult lr = matchLine(line);
if (!lr.href.isEmpty()) {
// Found something && line continuation
cursor.insertText(deferedText, charFormat(format));
deferedText.clear();
appendLine(cursor, lr, line, format);
} else {
// Found nothing, just emit the new part
deferedText += newPart;
}
// Handled line continuation
m_lastLine.clear();
} else {
const QString line = txt.mid(start, pos - start + 1);
LinkResult lr = matchLine(line);
if (!lr.href.isEmpty()) {
cursor.insertText(deferedText, charFormat(format));
deferedText.clear();
appendLine(cursor, lr, line, format);
deferedText += line;
}
}
start = pos + 1;
pos = txt.indexOf(QLatin1Char('\n'), start);
}
// Handle left over stuff
if (start < txt.length()) {
if (!m_lastLine.isEmpty()) {
// Line continuation
const QString newPart = txt.mid(start);
m_lastLine.append(newPart);
LinkResult lr = matchLine(m_lastLine);
if (!lr.href.isEmpty()) {
// Found something && line continuation
cursor.insertText(deferedText, charFormat(format));
deferedText.clear();
appendLine(cursor, lr, m_lastLine, format);
} else {
// Found nothing, just emit the new part
deferedText += newPart;
}
} else {
m_lastLine = txt.mid(start);
LinkResult lr = matchLine(m_lastLine);
if (!lr.href.isEmpty()) {
cursor.insertText(deferedText, charFormat(format));
deferedText.clear();
appendLine(cursor, lr, m_lastLine, format);
deferedText += m_lastLine;
cursor.insertText(deferedText, charFormat(format));
// deferedText.clear();
cursor.endEditBlock();
void QtOutputFormatter::appendLine(QTextCursor &cursor, LinkResult lr,
const QString &line, Utils::OutputFormat format)
const QTextCharFormat normalFormat = charFormat(format);
cursor.insertText(line.left(lr.start), normalFormat);
QTextCharFormat linkFormat = normalFormat;
const QColor textColor = plainTextEdit()->palette().color(QPalette::Text);
linkFormat.setForeground(mixColors(textColor, QColor(Qt::blue)));
linkFormat.setUnderlineStyle(QTextCharFormat::SingleUnderline);
linkFormat.setAnchor(true);
linkFormat.setAnchorHref(lr.href);
cursor.insertText(line.mid(lr.start, lr.end - lr.start), linkFormat);
cursor.insertText(line.mid(lr.end), normalFormat);
void QtOutputFormatter::handleLink(const QString &href)
{
if (!href.isEmpty()) {
const QRegExp qmlLineColumnLink(QLatin1String("^(file:///.+)" // file url
":(\\d+)" // line
":(\\d+)$")); // column
if (qmlLineColumnLink.indexIn(href) != -1) {
const QUrl fileUrl = QUrl(qmlLineColumnLink.cap(1));
const int line = qmlLineColumnLink.cap(2).toInt();
const int column = qmlLineColumnLink.cap(3).toInt();
TextEditor::BaseTextEditorWidget::openEditorAt(m_projectFinder.findFile(fileUrl), line, column - 1);
}
const QRegExp qmlLineLink(QLatin1String("^(file:///.+)" // file url
":(\\d+)$")); // line
if (qmlLineLink.indexIn(href) != -1) {
const QUrl fileUrl = QUrl(qmlLineLink.cap(1));
const int line = qmlLineLink.cap(2).toInt();
TextEditor::BaseTextEditorWidget::openEditorAt(m_projectFinder.findFile(fileUrl), line);
return;
}
QString fileName;
int line = -1;
if (qtErrorLink.indexIn(href) != -1) {
fileName = qtErrorLink.cap(1);
line = qtErrorLink.cap(2).toInt();
}
QRegExp qtAssertLink(QLatin1String("^(.+), line (\\d+)$"));
if (qtAssertLink.indexIn(href) != -1) {
fileName = qtAssertLink.cap(1);
line = qtAssertLink.cap(2).toInt();
}
QRegExp qtTestFailLink(QLatin1String("^(.*)\\((\\d+)\\)$"));
if (qtTestFailLink.indexIn(href) != -1) {
fileName = qtTestFailLink.cap(1);
line = qtTestFailLink.cap(2).toInt();
}
ProjectExplorer::Project *pro = m_project.data();
if (pro) {
QString baseName = fi.fileName();
foreach (const QString &file, pro->files(Project::AllFiles)) {
if (file.endsWith(baseName)) {
// pick the first one...
fileName = file;
break;
}
}
}

Kai Koehne
committed
} else if (!fi.exists()) {
// map possible on-device path to source path
fileName = m_projectFinder.findFile(QUrl::fromLocalFile(fileName));
TextEditor::BaseTextEditorWidget::openEditorAt(fileName, line, 0);
void QtOutputFormatter::updateProjectFileList()
{
if (m_project)
m_projectFinder.setProjectFiles(m_project.data()->files(Project::ExcludeGeneratedFiles));