/************************************************************************** ** ** This file is part of Qt Creator ** ** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies). ** ** Contact: Nokia Corporation (qt-info@nokia.com) ** ** No Commercial Usage ** ** This file contains pre-release code and may not be distributed. ** You may use this file in accordance with the terms and conditions ** contained in the Technology Preview License Agreement accompanying ** this package. ** ** GNU Lesser General Public License Usage ** ** Alternatively, 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. ** ** If you have questions regarding the use of this file, please contact ** Nokia at qt-info@nokia.com. ** **************************************************************************/ #include "qtoutputformatter.h" #include #include #include #include #include #include using namespace ProjectExplorer; using namespace Qt4ProjectManager; QtOutputFormatter::QtOutputFormatter(ProjectExplorer::Project *project) : OutputFormatter() , m_qmlError(QLatin1String("(file:///.+:\\d+:\\d+):")) , m_qtError(QLatin1String("Object::.*in (.*:\\d+)")) , m_qtAssert(QLatin1String("^ASSERT: .* in file (.+, line \\d+)$")) , m_qtTestFail(QLatin1String("^ Loc: \\[(.*)\\]$")) , m_project(project) { } 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_qtTestFail.indexIn(line) != -1) { lr.href = m_qtTestFail.cap(1); lr.start = m_qtTestFail.pos(1); lr.end = lr.start + lr.href.length(); } return lr; } void QtOutputFormatter::appendApplicationOutput(const QString &txt, bool onStdErr) { QTextCursor cursor(plainTextEdit()->document()); cursor.movePosition(QTextCursor::End); cursor.beginEditBlock(); QString text = txt; text.remove(QLatin1Char('\r')); 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, format(onStdErr ? StdErrFormat : StdOutFormat)); deferedText.clear(); clearLastLine(); appendLine(cursor, lr, line, onStdErr); } 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, format(onStdErr ? StdErrFormat : StdOutFormat)); deferedText.clear(); appendLine(cursor, lr, line, onStdErr); } else { 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, format(onStdErr ? StdErrFormat : StdOutFormat)); deferedText.clear(); clearLastLine(); appendLine(cursor, lr, m_lastLine, onStdErr); } 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, format(onStdErr ? StdErrFormat : StdOutFormat)); deferedText.clear(); appendLine(cursor, lr, m_lastLine, onStdErr); } else { deferedText += m_lastLine; } } } cursor.insertText(deferedText, format(onStdErr ? StdErrFormat : StdOutFormat)); // deferedText.clear(); cursor.endEditBlock(); } void QtOutputFormatter::appendLine(QTextCursor &cursor, LinkResult lr, const QString &line, bool onStdErr) { const QTextCharFormat normalFormat = format(onStdErr ? StdErrFormat : StdOutFormat); 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); } // Map absolute path in shadow build / in the deployment folder to the path in the project directory // // Input is e.g. // C:/app-build-desktop/qml/app/main.qml (shadow build directory) // C:/Private/e3026d63/qml/app/main.qml (Application data folder on Symbian device) // /Users/x/app-build-desktop/App.app/Contents/Resources/qml/App/main.qml (folder on Mac OS X) // which should be mapped to // $PROJECTDIR/qml/app/main.qml QString QtOutputFormatter::pathInSourceDirectory(const QString &originalFilePath) { QTC_ASSERT(QFileInfo(originalFilePath).isAbsolute(), return originalFilePath); if (!m_project) return originalFilePath; const QString projectDirectory = m_project.data()->projectDirectory(); QTC_ASSERT(!projectDirectory.isEmpty(), return originalFilePath); QTC_ASSERT(!projectDirectory.endsWith(QLatin1Char('/')), return originalFilePath); const QChar separator = QLatin1Char('/'); if (originalFilePath.startsWith(projectDirectory + separator)) { return originalFilePath; } // Strip directories one by one from the beginning of the path, // and see if the new relative path exists in the build directory. if (originalFilePath.contains(separator)) { for (int pos = originalFilePath.indexOf(separator); pos != -1; pos = originalFilePath.indexOf(separator, pos + 1)) { QString candidate = originalFilePath; candidate.remove(0, pos); candidate.prepend(projectDirectory); QFileInfo candidateInfo(candidate); if (candidateInfo.exists() && candidateInfo.isFile()) return candidate; } } return originalFilePath; } void QtOutputFormatter::handleLink(const QString &href) { if (!href.isEmpty()) { const QRegExp qmlErrorLink(QLatin1String("^(file:///.+):(\\d+):(\\d+)")); if (qmlErrorLink.indexIn(href) != -1) { const QString fileName = QUrl(qmlErrorLink.cap(1)).toLocalFile(); const int line = qmlErrorLink.cap(2).toInt(); const int column = qmlErrorLink.cap(3).toInt(); TextEditor::BaseTextEditor::openEditorAt(pathInSourceDirectory(fileName), line, column - 1); return; } QString fileName; int line = -1; QRegExp qtErrorLink(QLatin1String("^(.*):(\\d+)$")); 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(); } if (!fileName.isEmpty()) { QFileInfo fi(fileName); if (fi.isRelative()) { // Yeah fileName is relative, no suprise 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; } } } } else if (!fi.exists()) { // map possible on-device path to source path fileName = pathInSourceDirectory(fileName); } TextEditor::BaseTextEditor::openEditorAt(fileName, line, 0); return; } } }