2014-09-25 11:11:58 +02:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
2015-01-15 10:32:57 +01:00
|
|
|
** Copyright (C) 2015 The Qt Company Ltd
|
2014-09-25 11:11:58 +02:00
|
|
|
** All rights reserved.
|
2015-01-15 10:32:57 +01:00
|
|
|
** For any questions to The Qt Company, please use contact form at http://www.qt.io/contact-us
|
2014-09-25 11:11:58 +02:00
|
|
|
**
|
|
|
|
|
** This file is part of the Qt Enterprise LicenseChecker Add-on.
|
|
|
|
|
**
|
|
|
|
|
** Licensees holding valid Qt Enterprise licenses may use this file in
|
|
|
|
|
** accordance with the Qt Enterprise License Agreement provided with the
|
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
2015-01-15 10:32:57 +01:00
|
|
|
** a written agreement between you and The Qt Company.
|
2014-09-25 11:11:58 +02:00
|
|
|
**
|
|
|
|
|
** If you have questions regarding the use of this file, please use
|
2015-01-15 10:32:57 +01:00
|
|
|
** contact form at http://www.qt.io/contact-us
|
2014-09-25 11:11:58 +02:00
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include "clangstaticanalyzerdiagnosticmodel.h"
|
|
|
|
|
|
2015-02-19 18:08:38 +01:00
|
|
|
#include "clangstaticanalyzerprojectsettingsmanager.h"
|
2014-09-25 11:11:58 +02:00
|
|
|
#include "clangstaticanalyzerutils.h"
|
|
|
|
|
|
2015-02-19 18:08:38 +01:00
|
|
|
#include <projectexplorer/project.h>
|
|
|
|
|
#include <projectexplorer/session.h>
|
|
|
|
|
#include <utils/qtcassert.h>
|
|
|
|
|
|
2014-09-25 11:11:58 +02:00
|
|
|
#include <QCoreApplication>
|
2015-02-19 18:08:38 +01:00
|
|
|
#include <QFileInfo>
|
2014-09-25 11:11:58 +02:00
|
|
|
|
|
|
|
|
namespace ClangStaticAnalyzer {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
|
|
ClangStaticAnalyzerDiagnosticModel::ClangStaticAnalyzerDiagnosticModel(QObject *parent)
|
|
|
|
|
: QAbstractListModel(parent)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangStaticAnalyzerDiagnosticModel::addDiagnostics(const QList<Diagnostic> &diagnostics)
|
|
|
|
|
{
|
|
|
|
|
beginInsertRows(QModelIndex(), m_diagnostics.size(),
|
|
|
|
|
m_diagnostics.size() + diagnostics.size() - 1 );
|
|
|
|
|
m_diagnostics += diagnostics;
|
|
|
|
|
endInsertRows();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangStaticAnalyzerDiagnosticModel::clear()
|
|
|
|
|
{
|
|
|
|
|
beginResetModel();
|
|
|
|
|
m_diagnostics.clear();
|
|
|
|
|
endResetModel();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int ClangStaticAnalyzerDiagnosticModel::rowCount(const QModelIndex &parent) const
|
|
|
|
|
{
|
|
|
|
|
return parent.isValid() ? 0 : m_diagnostics.count();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static QString createDiagnosticToolTipString(const Diagnostic &diagnostic)
|
|
|
|
|
{
|
|
|
|
|
typedef QPair<QString, QString> StringPair;
|
|
|
|
|
QList<StringPair> lines;
|
|
|
|
|
|
|
|
|
|
if (!diagnostic.category.isEmpty()) {
|
|
|
|
|
lines << qMakePair(
|
|
|
|
|
QCoreApplication::translate("ClangStaticAnalyzer::Diagnostic", "Category:"),
|
2014-11-14 16:32:36 +01:00
|
|
|
diagnostic.category.toHtmlEscaped());
|
2014-09-25 11:11:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!diagnostic.type.isEmpty()) {
|
|
|
|
|
lines << qMakePair(
|
|
|
|
|
QCoreApplication::translate("ClangStaticAnalyzer::Diagnostic", "Type:"),
|
2014-11-14 16:32:36 +01:00
|
|
|
diagnostic.type.toHtmlEscaped());
|
2014-09-25 11:11:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!diagnostic.issueContext.isEmpty() && !diagnostic.issueContextKind.isEmpty()) {
|
|
|
|
|
lines << qMakePair(
|
|
|
|
|
QCoreApplication::translate("ClangStaticAnalyzer::Diagnostic", "Context:"),
|
2014-11-14 16:32:36 +01:00
|
|
|
diagnostic.issueContextKind.toHtmlEscaped() + QLatin1Char(' ')
|
|
|
|
|
+ diagnostic.issueContext.toHtmlEscaped());
|
2014-09-25 11:11:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
lines << qMakePair(
|
|
|
|
|
QCoreApplication::translate("ClangStaticAnalyzer::Diagnostic", "Location:"),
|
|
|
|
|
createFullLocationString(diagnostic.location));
|
|
|
|
|
|
|
|
|
|
QString html = QLatin1String("<html>"
|
|
|
|
|
"<head>"
|
|
|
|
|
"<style>dt { font-weight:bold; } dd { font-family: monospace; }</style>\n"
|
|
|
|
|
"<body><dl>");
|
|
|
|
|
|
|
|
|
|
foreach (const StringPair &pair, lines) {
|
|
|
|
|
html += QLatin1String("<dt>");
|
|
|
|
|
html += pair.first;
|
|
|
|
|
html += QLatin1String("</dt><dd>");
|
|
|
|
|
html += pair.second;
|
|
|
|
|
html += QLatin1String("</dd>\n");
|
|
|
|
|
}
|
|
|
|
|
html += QLatin1String("</dl></body></html>");
|
|
|
|
|
return html;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QVariant ClangStaticAnalyzerDiagnosticModel::data(const QModelIndex &index, int role) const
|
|
|
|
|
{
|
|
|
|
|
if (!index.isValid())
|
|
|
|
|
return QVariant();
|
|
|
|
|
|
|
|
|
|
if (index.parent().isValid())
|
|
|
|
|
return QVariant();
|
|
|
|
|
|
|
|
|
|
const int row = index.row();
|
|
|
|
|
if (row < 0 || row >= m_diagnostics.size())
|
|
|
|
|
return QVariant();
|
|
|
|
|
|
|
|
|
|
const Diagnostic diagnostic = m_diagnostics.at(row);
|
|
|
|
|
|
|
|
|
|
if (role == Qt::DisplayRole)
|
|
|
|
|
return QString(QLatin1String("Some specific diagnostic")); // TODO: Remove?
|
|
|
|
|
else if (role == Qt::ToolTipRole)
|
|
|
|
|
return createDiagnosticToolTipString(diagnostic);
|
|
|
|
|
else if (role == Qt::UserRole)
|
|
|
|
|
return QVariant::fromValue<Diagnostic>(diagnostic);
|
|
|
|
|
|
|
|
|
|
return QVariant();
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-19 18:08:38 +01:00
|
|
|
|
|
|
|
|
ClangStaticAnalyzerDiagnosticFilterModel::ClangStaticAnalyzerDiagnosticFilterModel(QObject *parent)
|
|
|
|
|
: QSortFilterProxyModel(parent)
|
|
|
|
|
{
|
|
|
|
|
// So that when a user closes and re-opens a project and *then* clicks "Suppress",
|
|
|
|
|
// we enter that information into the project settings.
|
|
|
|
|
connect(ProjectExplorer::SessionManager::instance(),
|
|
|
|
|
&ProjectExplorer::SessionManager::projectAdded, this,
|
|
|
|
|
[this](ProjectExplorer::Project *project) {
|
|
|
|
|
if (!m_project && project->projectDirectory() == m_lastProjectDirectory)
|
|
|
|
|
setProject(project);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangStaticAnalyzerDiagnosticFilterModel::setProject(ProjectExplorer::Project *project)
|
|
|
|
|
{
|
|
|
|
|
QTC_ASSERT(project, return);
|
|
|
|
|
if (m_project) {
|
|
|
|
|
disconnect(ProjectSettingsManager::getSettings(m_project),
|
|
|
|
|
&ProjectSettings::suppressedDiagnosticsChanged, this,
|
|
|
|
|
&ClangStaticAnalyzerDiagnosticFilterModel::handleSuppressedDiagnosticsChanged);
|
|
|
|
|
}
|
|
|
|
|
m_project = project;
|
|
|
|
|
m_lastProjectDirectory = m_project->projectDirectory();
|
|
|
|
|
connect(ProjectSettingsManager::getSettings(m_project),
|
|
|
|
|
&ProjectSettings::suppressedDiagnosticsChanged,
|
|
|
|
|
this, &ClangStaticAnalyzerDiagnosticFilterModel::handleSuppressedDiagnosticsChanged);
|
|
|
|
|
handleSuppressedDiagnosticsChanged();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangStaticAnalyzerDiagnosticFilterModel::addSuppressedDiagnostic(
|
|
|
|
|
const SuppressedDiagnostic &diag)
|
|
|
|
|
{
|
|
|
|
|
QTC_ASSERT(!m_project, return);
|
|
|
|
|
m_suppressedDiagnostics << diag;
|
|
|
|
|
invalidate();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool ClangStaticAnalyzerDiagnosticFilterModel::filterAcceptsRow(int sourceRow,
|
|
|
|
|
const QModelIndex &sourceParent) const
|
|
|
|
|
{
|
|
|
|
|
Q_UNUSED(sourceParent);
|
|
|
|
|
const Diagnostic diag = static_cast<ClangStaticAnalyzerDiagnosticModel *>(sourceModel())
|
|
|
|
|
->diagnostics().at(sourceRow);
|
|
|
|
|
foreach (const SuppressedDiagnostic &d, m_suppressedDiagnostics) {
|
|
|
|
|
if (d.description != diag.description)
|
|
|
|
|
continue;
|
|
|
|
|
QString filePath = d.filePath.toString();
|
|
|
|
|
QFileInfo fi(filePath);
|
|
|
|
|
if (fi.isRelative())
|
|
|
|
|
filePath = m_lastProjectDirectory.toString() + QLatin1Char('/') + filePath;
|
|
|
|
|
if (filePath == diag.location.filePath)
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangStaticAnalyzerDiagnosticFilterModel::handleSuppressedDiagnosticsChanged()
|
|
|
|
|
{
|
|
|
|
|
QTC_ASSERT(m_project, return);
|
|
|
|
|
m_suppressedDiagnostics
|
|
|
|
|
= ProjectSettingsManager::getSettings(m_project)->suppressedDiagnostics();
|
|
|
|
|
invalidate();
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-25 11:11:58 +02:00
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace ClangStaticAnalyzer
|