2016-02-22 17:18:18 +01:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
|
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator.
|
|
|
|
|
**
|
|
|
|
|
** Commercial License Usage
|
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
|
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
|
|
|
|
**
|
|
|
|
|
** GNU General Public License Usage
|
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
|
** General Public License version 3 as published by the Free Software
|
|
|
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
|
|
|
** included in the packaging of this file. Please review the following
|
|
|
|
|
** information to ensure the GNU General Public License requirements will
|
|
|
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
|
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include "clangdiagnosticconfigswidget.h"
|
2018-05-04 15:58:41 +02:00
|
|
|
|
|
|
|
|
#include "cppcodemodelsettings.h"
|
2018-05-16 07:58:53 +02:00
|
|
|
#include "cpptools_clangtidychecks.h"
|
2018-05-29 14:38:57 +02:00
|
|
|
#include "cpptoolsconstants.h"
|
2018-05-04 15:58:41 +02:00
|
|
|
#include "cpptoolsreuse.h"
|
2016-02-22 17:18:18 +01:00
|
|
|
#include "ui_clangdiagnosticconfigswidget.h"
|
2018-02-06 13:17:36 +01:00
|
|
|
#include "ui_clangbasechecks.h"
|
2018-01-26 10:27:58 +01:00
|
|
|
#include "ui_clazychecks.h"
|
|
|
|
|
#include "ui_tidychecks.h"
|
2016-02-22 17:18:18 +01:00
|
|
|
|
2018-05-16 07:58:53 +02:00
|
|
|
#include <projectexplorer/selectablefilesmodel.h>
|
|
|
|
|
|
2016-02-22 17:18:18 +01:00
|
|
|
#include <utils/algorithm.h>
|
|
|
|
|
#include <utils/qtcassert.h>
|
2017-09-08 13:00:44 +02:00
|
|
|
#include <utils/utilsicons.h>
|
2016-02-22 17:18:18 +01:00
|
|
|
|
|
|
|
|
#include <QDebug>
|
2018-05-29 14:38:57 +02:00
|
|
|
#include <QDesktopServices>
|
2018-05-16 07:58:53 +02:00
|
|
|
#include <QDialogButtonBox>
|
2016-02-22 17:18:18 +01:00
|
|
|
#include <QInputDialog>
|
2018-05-04 15:58:41 +02:00
|
|
|
#include <QPushButton>
|
2016-02-22 17:18:18 +01:00
|
|
|
#include <QUuid>
|
|
|
|
|
|
|
|
|
|
namespace CppTools {
|
|
|
|
|
|
2018-05-29 14:38:57 +02:00
|
|
|
static constexpr const char CLANG_STATIC_ANALYZER_URL[]
|
|
|
|
|
= "https://clang-analyzer.llvm.org/available_checks.html";
|
|
|
|
|
|
2018-05-16 07:58:53 +02:00
|
|
|
static void buildTree(ProjectExplorer::Tree *parent,
|
|
|
|
|
ProjectExplorer::Tree *current,
|
|
|
|
|
const Constants::TidyNode &node)
|
|
|
|
|
{
|
|
|
|
|
current->name = QString::fromUtf8(node.name);
|
|
|
|
|
current->isDir = node.children.size();
|
|
|
|
|
if (parent) {
|
|
|
|
|
current->fullPath = parent->fullPath + current->name;
|
|
|
|
|
parent->childDirectories.push_back(current);
|
|
|
|
|
} else {
|
|
|
|
|
current->fullPath = Utils::FileName::fromString(current->name);
|
|
|
|
|
}
|
|
|
|
|
current->parent = parent;
|
|
|
|
|
for (const Constants::TidyNode &nodeChild : node.children)
|
|
|
|
|
buildTree(current, new ProjectExplorer::Tree, nodeChild);
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-29 14:38:57 +02:00
|
|
|
static bool needsLink(ProjectExplorer::Tree *node) {
|
|
|
|
|
if (node->name == "clang-analyzer-")
|
|
|
|
|
return true;
|
|
|
|
|
return !node->isDir && !node->fullPath.toString().startsWith("clang-analyzer-");
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-16 07:58:53 +02:00
|
|
|
class TidyChecksTreeModel final : public ProjectExplorer::SelectableFilesModel
|
|
|
|
|
{
|
|
|
|
|
Q_OBJECT
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
TidyChecksTreeModel()
|
|
|
|
|
: ProjectExplorer::SelectableFilesModel(nullptr)
|
|
|
|
|
{
|
|
|
|
|
buildTree(nullptr, m_root, Constants::CLANG_TIDY_CHECKS_ROOT);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString selectedChecks() const
|
|
|
|
|
{
|
|
|
|
|
QString checks;
|
|
|
|
|
collectChecks(m_root, checks);
|
|
|
|
|
return "-*" + checks;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void selectChecks(const QString &checks)
|
|
|
|
|
{
|
|
|
|
|
m_root->checked = Qt::Unchecked;
|
|
|
|
|
propagateDown(index(0, 0, QModelIndex()));
|
|
|
|
|
|
|
|
|
|
QStringList checksList = checks.simplified().remove(" ")
|
|
|
|
|
.split(",", QString::SkipEmptyParts);
|
|
|
|
|
|
|
|
|
|
for (QString &check : checksList) {
|
|
|
|
|
Qt::CheckState state;
|
|
|
|
|
if (check.startsWith("-")) {
|
|
|
|
|
check = check.right(check.length() - 1);
|
|
|
|
|
state = Qt::Unchecked;
|
|
|
|
|
} else {
|
|
|
|
|
state = Qt::Checked;
|
|
|
|
|
}
|
|
|
|
|
const QModelIndex index = indexForCheck(check);
|
|
|
|
|
if (!index.isValid())
|
|
|
|
|
continue;
|
|
|
|
|
auto node = static_cast<ProjectExplorer::Tree *>(index.internalPointer());
|
|
|
|
|
node->checked = state;
|
|
|
|
|
propagateUp(index);
|
|
|
|
|
propagateDown(index);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-29 14:38:57 +02:00
|
|
|
int columnCount(const QModelIndex &/*parent*/) const override
|
|
|
|
|
{
|
|
|
|
|
return 2;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QVariant data(const QModelIndex &fullIndex, int role = Qt::DisplayRole) const final
|
2018-05-16 07:58:53 +02:00
|
|
|
{
|
2018-05-29 14:38:57 +02:00
|
|
|
if (!fullIndex.isValid() || role == Qt::DecorationRole)
|
|
|
|
|
return QVariant();
|
|
|
|
|
QModelIndex index = this->index(fullIndex.row(), 0, fullIndex.parent());
|
|
|
|
|
auto *node = static_cast<ProjectExplorer::Tree *>(index.internalPointer());
|
|
|
|
|
|
|
|
|
|
if (fullIndex.column() == 1) {
|
|
|
|
|
if (!needsLink(node))
|
|
|
|
|
return QVariant();
|
|
|
|
|
switch (role) {
|
|
|
|
|
case Qt::DisplayRole:
|
|
|
|
|
return tr("Web Page");
|
|
|
|
|
case Qt::FontRole: {
|
|
|
|
|
QFont font = QApplication::font();
|
|
|
|
|
font.setUnderline(true);
|
|
|
|
|
return font;
|
|
|
|
|
}
|
|
|
|
|
case Qt::ForegroundRole:
|
|
|
|
|
return QApplication::palette().link().color();
|
|
|
|
|
case Qt::UserRole: {
|
|
|
|
|
// 'clang-analyzer-' group
|
|
|
|
|
if (node->isDir)
|
|
|
|
|
return QString::fromUtf8(CLANG_STATIC_ANALYZER_URL);
|
|
|
|
|
return QString::fromUtf8(Constants::TIDY_DOCUMENTATION_URL_TEMPLATE)
|
|
|
|
|
.arg(node->fullPath.toString());
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-05-16 07:58:53 +02:00
|
|
|
return QVariant();
|
|
|
|
|
}
|
2018-05-29 14:38:57 +02:00
|
|
|
|
|
|
|
|
if (role == Qt::DisplayRole)
|
|
|
|
|
return node->isDir ? (node->name + "*") : node->name;
|
|
|
|
|
|
2018-05-16 07:58:53 +02:00
|
|
|
return ProjectExplorer::SelectableFilesModel::data(index, role);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
|
|
// TODO: Remove/replace this method after base class refactoring is done.
|
|
|
|
|
void traverse(const QModelIndex &index,
|
|
|
|
|
const std::function<bool(const QModelIndex &)> &visit) const
|
|
|
|
|
{
|
|
|
|
|
if (!index.isValid())
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (!visit(index))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (!hasChildren(index))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
const int rows = rowCount(index);
|
|
|
|
|
const int cols = columnCount(index);
|
|
|
|
|
for (int i = 0; i < rows; ++i) {
|
|
|
|
|
for (int j = 0; j < cols; ++j)
|
|
|
|
|
traverse(this->index(i, j, index), visit);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QModelIndex indexForCheck(const QString &check) const {
|
|
|
|
|
if (check == "*")
|
|
|
|
|
return index(0, 0, QModelIndex());
|
|
|
|
|
|
|
|
|
|
QModelIndex result;
|
|
|
|
|
traverse(index(0, 0, QModelIndex()), [&](const QModelIndex &index){
|
|
|
|
|
using ProjectExplorer::Tree;
|
|
|
|
|
if (result.isValid())
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
auto *node = static_cast<Tree *>(index.internalPointer());
|
|
|
|
|
const QString nodeName = node->fullPath.toString();
|
|
|
|
|
if ((check.endsWith("*") && nodeName.startsWith(check.left(check.length() - 1)))
|
|
|
|
|
|| (!node->isDir && nodeName == check)) {
|
|
|
|
|
result = index;
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-29 14:38:57 +02:00
|
|
|
return check.startsWith(nodeName);
|
2018-05-16 07:58:53 +02:00
|
|
|
});
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void collectChecks(const ProjectExplorer::Tree *root, QString &checks)
|
|
|
|
|
{
|
|
|
|
|
if (root->checked == Qt::Unchecked)
|
|
|
|
|
return;
|
|
|
|
|
if (root->checked == Qt::Checked) {
|
|
|
|
|
checks += "," + root->fullPath.toString();
|
|
|
|
|
if (root->isDir)
|
|
|
|
|
checks += "*";
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
for (const ProjectExplorer::Tree *t : root->childDirectories)
|
|
|
|
|
collectChecks(t, checks);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2018-05-22 16:22:53 +02:00
|
|
|
ClangDiagnosticConfigsWidget::ClangDiagnosticConfigsWidget(const Core::Id &configToSelect,
|
|
|
|
|
QWidget *parent)
|
2016-02-22 17:18:18 +01:00
|
|
|
: QWidget(parent)
|
|
|
|
|
, m_ui(new Ui::ClangDiagnosticConfigsWidget)
|
2018-05-04 15:58:41 +02:00
|
|
|
, m_diagnosticConfigsModel(codeModelSettings()->clangCustomDiagnosticConfigs())
|
2018-05-16 07:58:53 +02:00
|
|
|
, m_tidyTreeModel(new TidyChecksTreeModel())
|
2016-02-22 17:18:18 +01:00
|
|
|
{
|
|
|
|
|
m_ui->setupUi(this);
|
2018-02-06 13:17:36 +01:00
|
|
|
setupTabs();
|
2016-02-22 17:18:18 +01:00
|
|
|
|
2018-05-04 15:58:41 +02:00
|
|
|
m_selectedConfigIndex = m_diagnosticConfigsModel.indexOfConfig(
|
|
|
|
|
codeModelSettings()->clangDiagnosticConfigId());
|
|
|
|
|
|
2016-02-26 17:50:38 +01:00
|
|
|
connectConfigChooserCurrentIndex();
|
2016-02-22 17:18:18 +01:00
|
|
|
connect(m_ui->copyButton, &QPushButton::clicked,
|
|
|
|
|
this, &ClangDiagnosticConfigsWidget::onCopyButtonClicked);
|
|
|
|
|
connect(m_ui->removeButton, &QPushButton::clicked,
|
|
|
|
|
this, &ClangDiagnosticConfigsWidget::onRemoveButtonClicked);
|
2016-02-26 17:50:38 +01:00
|
|
|
connectDiagnosticOptionsChanged();
|
2016-02-22 17:18:18 +01:00
|
|
|
|
2018-05-29 14:38:57 +02:00
|
|
|
connect(m_tidyChecks->checksPrefixesTree, &QTreeView::clicked,
|
|
|
|
|
[this](const QModelIndex &index) {
|
|
|
|
|
const QString link = m_tidyTreeModel->data(index, Qt::UserRole).toString();
|
|
|
|
|
if (link.isEmpty())
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
QDesktopServices::openUrl(QUrl(link));
|
|
|
|
|
});
|
|
|
|
|
|
2018-05-22 16:22:53 +02:00
|
|
|
syncWidgetsToModel(configToSelect);
|
2016-02-22 17:18:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ClangDiagnosticConfigsWidget::~ClangDiagnosticConfigsWidget()
|
|
|
|
|
{
|
|
|
|
|
delete m_ui;
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-04 15:58:41 +02:00
|
|
|
void ClangDiagnosticConfigsWidget::onCurrentConfigChanged(int index)
|
2016-02-22 17:18:18 +01:00
|
|
|
{
|
2018-05-04 15:58:41 +02:00
|
|
|
m_selectedConfigIndex = index;
|
2016-02-22 17:18:18 +01:00
|
|
|
syncOtherWidgetsToComboBox();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static ClangDiagnosticConfig createCustomConfig(const ClangDiagnosticConfig &config,
|
|
|
|
|
const QString &displayName)
|
|
|
|
|
{
|
|
|
|
|
ClangDiagnosticConfig copied = config;
|
|
|
|
|
copied.setId(Core::Id::fromString(QUuid::createUuid().toString()));
|
|
|
|
|
copied.setDisplayName(displayName);
|
|
|
|
|
copied.setIsReadOnly(false);
|
|
|
|
|
|
|
|
|
|
return copied;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangDiagnosticConfigsWidget::onCopyButtonClicked()
|
|
|
|
|
{
|
2018-05-04 15:58:41 +02:00
|
|
|
const ClangDiagnosticConfig &config = selectedConfig();
|
2016-02-22 17:18:18 +01:00
|
|
|
|
|
|
|
|
bool diaglogAccepted = false;
|
|
|
|
|
const QString newName = QInputDialog::getText(this,
|
|
|
|
|
tr("Copy Diagnostic Configuration"),
|
|
|
|
|
tr("Diagnostic configuration name:"),
|
|
|
|
|
QLineEdit::Normal,
|
|
|
|
|
tr("%1 (Copy)").arg(config.displayName()),
|
|
|
|
|
&diaglogAccepted);
|
|
|
|
|
if (diaglogAccepted) {
|
|
|
|
|
const ClangDiagnosticConfig customConfig = createCustomConfig(config, newName);
|
|
|
|
|
m_diagnosticConfigsModel.appendOrUpdate(customConfig);
|
2016-02-26 17:50:38 +01:00
|
|
|
emit customConfigsChanged(customConfigs());
|
2016-02-22 17:18:18 +01:00
|
|
|
|
|
|
|
|
syncConfigChooserToModel(customConfig.id());
|
2018-02-06 13:17:36 +01:00
|
|
|
m_clangBaseChecks->diagnosticOptionsTextEdit->setFocus();
|
2016-02-22 17:18:18 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-04 15:58:41 +02:00
|
|
|
const ClangDiagnosticConfig &ClangDiagnosticConfigsWidget::selectedConfig() const
|
|
|
|
|
{
|
|
|
|
|
return m_diagnosticConfigsModel.at(m_selectedConfigIndex);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Core::Id ClangDiagnosticConfigsWidget::selectedConfigId() const
|
|
|
|
|
{
|
|
|
|
|
return selectedConfig().id();
|
|
|
|
|
}
|
|
|
|
|
|
2016-02-22 17:18:18 +01:00
|
|
|
void ClangDiagnosticConfigsWidget::onRemoveButtonClicked()
|
|
|
|
|
{
|
2018-05-04 15:58:41 +02:00
|
|
|
m_diagnosticConfigsModel.removeConfigWithId(selectedConfigId());
|
2016-02-26 17:50:38 +01:00
|
|
|
emit customConfigsChanged(customConfigs());
|
2016-02-22 17:18:18 +01:00
|
|
|
|
|
|
|
|
syncConfigChooserToModel();
|
|
|
|
|
}
|
|
|
|
|
|
2018-02-23 11:36:18 +01:00
|
|
|
void ClangDiagnosticConfigsWidget::onClangTidyModeChanged(int index)
|
|
|
|
|
{
|
2018-05-04 15:58:41 +02:00
|
|
|
ClangDiagnosticConfig config = selectedConfig();
|
2018-02-23 11:36:18 +01:00
|
|
|
config.setClangTidyMode(static_cast<ClangDiagnosticConfig::TidyMode>(index));
|
|
|
|
|
updateConfig(config);
|
|
|
|
|
syncClangTidyWidgets(config);
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-16 07:58:53 +02:00
|
|
|
void ClangDiagnosticConfigsWidget::onClangTidyTreeChanged()
|
2018-01-26 10:27:58 +01:00
|
|
|
{
|
2018-05-04 15:58:41 +02:00
|
|
|
ClangDiagnosticConfig config = selectedConfig();
|
2018-05-16 07:58:53 +02:00
|
|
|
config.setClangTidyChecks(m_tidyTreeModel->selectedChecks());
|
2018-01-26 10:27:58 +01:00
|
|
|
updateConfig(config);
|
|
|
|
|
}
|
|
|
|
|
|
2018-02-07 09:20:30 +01:00
|
|
|
void ClangDiagnosticConfigsWidget::onClazyRadioButtonChanged(bool checked)
|
|
|
|
|
{
|
|
|
|
|
if (!checked)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
QString checks;
|
|
|
|
|
if (m_clazyChecks->clazyRadioDisabled->isChecked())
|
|
|
|
|
checks = QString();
|
|
|
|
|
else if (m_clazyChecks->clazyRadioLevel0->isChecked())
|
|
|
|
|
checks = "level0";
|
|
|
|
|
else if (m_clazyChecks->clazyRadioLevel1->isChecked())
|
|
|
|
|
checks = "level1";
|
|
|
|
|
else if (m_clazyChecks->clazyRadioLevel2->isChecked())
|
|
|
|
|
checks = "level2";
|
|
|
|
|
else if (m_clazyChecks->clazyRadioLevel3->isChecked())
|
|
|
|
|
checks = "level3";
|
|
|
|
|
|
2018-05-04 15:58:41 +02:00
|
|
|
ClangDiagnosticConfig config = selectedConfig();
|
2018-02-07 09:20:30 +01:00
|
|
|
config.setClazyChecks(checks);
|
|
|
|
|
updateConfig(config);
|
|
|
|
|
}
|
|
|
|
|
|
2017-09-08 13:00:44 +02:00
|
|
|
static bool isAcceptedWarningOption(const QString &option)
|
|
|
|
|
{
|
|
|
|
|
return option == "-w"
|
|
|
|
|
|| option == "-pedantic"
|
|
|
|
|
|| option == "-pedantic-errors";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Reference:
|
|
|
|
|
// https://gcc.gnu.org/onlinedocs/gcc/Warning-Options.html
|
|
|
|
|
// https://clang.llvm.org/docs/DiagnosticsReference.html
|
|
|
|
|
static bool isValidOption(const QString &option)
|
|
|
|
|
{
|
|
|
|
|
if (option == "-Werror")
|
|
|
|
|
return false; // Avoid errors due to unknown or misspelled warnings.
|
|
|
|
|
return option.startsWith("-W") || isAcceptedWarningOption(option);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static QString validateDiagnosticOptions(const QStringList &options)
|
|
|
|
|
{
|
|
|
|
|
// This is handy for testing, allow disabling validation.
|
|
|
|
|
if (qEnvironmentVariableIntValue("QTC_CLANG_NO_DIAGNOSTIC_CHECK"))
|
|
|
|
|
return QString();
|
|
|
|
|
|
|
|
|
|
for (const QString &option : options) {
|
|
|
|
|
if (!isValidOption(option))
|
|
|
|
|
return ClangDiagnosticConfigsWidget::tr("Option \"%1\" is invalid.").arg(option);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return QString();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static QStringList normalizeDiagnosticInputOptions(const QString &options)
|
|
|
|
|
{
|
|
|
|
|
return options.simplified().split(QLatin1Char(' '), QString::SkipEmptyParts);
|
|
|
|
|
}
|
|
|
|
|
|
2016-02-22 17:18:18 +01:00
|
|
|
void ClangDiagnosticConfigsWidget::onDiagnosticOptionsEdited()
|
|
|
|
|
{
|
2017-09-08 13:00:44 +02:00
|
|
|
// Clean up input
|
2018-02-06 13:17:36 +01:00
|
|
|
const QString diagnosticOptions = m_clangBaseChecks->diagnosticOptionsTextEdit->document()
|
|
|
|
|
->toPlainText();
|
2017-09-08 13:00:44 +02:00
|
|
|
const QStringList normalizedOptions = normalizeDiagnosticInputOptions(diagnosticOptions);
|
|
|
|
|
|
|
|
|
|
// Validate
|
|
|
|
|
const QString errorMessage = validateDiagnosticOptions(normalizedOptions);
|
|
|
|
|
updateValidityWidgets(errorMessage);
|
|
|
|
|
if (!errorMessage.isEmpty()) {
|
|
|
|
|
// Remember the entered options in case the user will switch back.
|
2018-05-04 15:58:41 +02:00
|
|
|
m_notAcceptedOptions.insert(selectedConfigId(), diagnosticOptions);
|
2017-09-08 13:00:44 +02:00
|
|
|
return;
|
|
|
|
|
}
|
2018-05-04 15:58:41 +02:00
|
|
|
m_notAcceptedOptions.remove(selectedConfigId());
|
2016-02-22 17:18:18 +01:00
|
|
|
|
2017-09-08 13:00:44 +02:00
|
|
|
// Commit valid changes
|
2018-05-04 15:58:41 +02:00
|
|
|
ClangDiagnosticConfig updatedConfig = selectedConfig();
|
2018-01-26 10:27:58 +01:00
|
|
|
updatedConfig.setClangOptions(normalizedOptions);
|
|
|
|
|
updateConfig(updatedConfig);
|
2016-02-22 17:18:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangDiagnosticConfigsWidget::syncWidgetsToModel(const Core::Id &configToSelect)
|
|
|
|
|
{
|
|
|
|
|
syncConfigChooserToModel(configToSelect);
|
|
|
|
|
syncOtherWidgetsToComboBox();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangDiagnosticConfigsWidget::syncConfigChooserToModel(const Core::Id &configToSelect)
|
|
|
|
|
{
|
2016-02-26 17:50:38 +01:00
|
|
|
disconnectConfigChooserCurrentIndex();
|
|
|
|
|
|
2018-05-11 14:53:17 +02:00
|
|
|
m_ui->configChooserList->clear();
|
2018-05-04 15:58:41 +02:00
|
|
|
m_selectedConfigIndex = std::max(std::min(m_selectedConfigIndex,
|
|
|
|
|
m_diagnosticConfigsModel.size() - 1),
|
|
|
|
|
0);
|
2016-02-22 17:18:18 +01:00
|
|
|
|
|
|
|
|
const int size = m_diagnosticConfigsModel.size();
|
|
|
|
|
for (int i = 0; i < size; ++i) {
|
|
|
|
|
const ClangDiagnosticConfig &config = m_diagnosticConfigsModel.at(i);
|
2016-02-26 17:50:38 +01:00
|
|
|
const QString displayName
|
2017-05-23 09:49:22 +02:00
|
|
|
= ClangDiagnosticConfigsModel::displayNameWithBuiltinIndication(config);
|
2018-05-11 14:53:17 +02:00
|
|
|
m_ui->configChooserList->addItem(displayName);
|
2016-02-22 17:18:18 +01:00
|
|
|
|
|
|
|
|
if (configToSelect == config.id())
|
2018-05-04 15:58:41 +02:00
|
|
|
m_selectedConfigIndex = i;
|
2016-02-22 17:18:18 +01:00
|
|
|
}
|
|
|
|
|
|
2016-02-26 17:50:38 +01:00
|
|
|
connectConfigChooserCurrentIndex();
|
|
|
|
|
|
2018-05-11 14:53:17 +02:00
|
|
|
m_ui->configChooserList->setCurrentRow(m_selectedConfigIndex);
|
2016-02-22 17:18:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangDiagnosticConfigsWidget::syncOtherWidgetsToComboBox()
|
|
|
|
|
{
|
|
|
|
|
if (isConfigChooserEmpty())
|
|
|
|
|
return;
|
|
|
|
|
|
2018-05-04 15:58:41 +02:00
|
|
|
const ClangDiagnosticConfig &config = selectedConfig();
|
2016-02-22 17:18:18 +01:00
|
|
|
|
|
|
|
|
// Update main button row
|
|
|
|
|
m_ui->removeButton->setEnabled(!config.isReadOnly());
|
|
|
|
|
|
2018-01-26 10:27:58 +01:00
|
|
|
// Update Text Edit
|
2017-09-08 13:00:44 +02:00
|
|
|
const QString options = m_notAcceptedOptions.contains(config.id())
|
|
|
|
|
? m_notAcceptedOptions.value(config.id())
|
2018-01-26 10:27:58 +01:00
|
|
|
: config.clangOptions().join(QLatin1Char(' '));
|
2017-09-08 13:00:44 +02:00
|
|
|
setDiagnosticOptions(options);
|
2018-02-12 12:04:08 +01:00
|
|
|
m_clangBaseChecksWidget->setEnabled(!config.isReadOnly());
|
2018-01-26 10:27:58 +01:00
|
|
|
|
2018-02-12 11:34:38 +01:00
|
|
|
if (config.isReadOnly()) {
|
|
|
|
|
m_ui->infoIcon->setPixmap(Utils::Icons::INFO.pixmap());
|
|
|
|
|
m_ui->infoLabel->setText(tr("Copy this configuration to customize it."));
|
|
|
|
|
m_ui->infoLabel->setStyleSheet(QString());
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-26 10:27:58 +01:00
|
|
|
syncClangTidyWidgets(config);
|
|
|
|
|
syncClazyWidgets(config);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangDiagnosticConfigsWidget::syncClangTidyWidgets(const ClangDiagnosticConfig &config)
|
|
|
|
|
{
|
|
|
|
|
disconnectClangTidyItemChanged();
|
|
|
|
|
|
2018-02-23 11:36:18 +01:00
|
|
|
ClangDiagnosticConfig::TidyMode tidyMode = config.clangTidyMode();
|
|
|
|
|
|
|
|
|
|
m_tidyChecks->tidyMode->setCurrentIndex(static_cast<int>(tidyMode));
|
|
|
|
|
switch (tidyMode) {
|
|
|
|
|
case ClangDiagnosticConfig::TidyMode::Disabled:
|
|
|
|
|
case ClangDiagnosticConfig::TidyMode::File:
|
2018-05-16 07:58:53 +02:00
|
|
|
m_tidyChecks->plainTextEditButton->setVisible(false);
|
2018-02-23 11:36:18 +01:00
|
|
|
m_tidyChecks->checksListWrapper->setCurrentIndex(1);
|
|
|
|
|
break;
|
|
|
|
|
case ClangDiagnosticConfig::TidyMode::ChecksPrefixList:
|
2018-05-16 07:58:53 +02:00
|
|
|
m_tidyChecks->plainTextEditButton->setVisible(true);
|
2018-02-23 11:36:18 +01:00
|
|
|
m_tidyChecks->checksListWrapper->setCurrentIndex(0);
|
2018-05-16 07:58:53 +02:00
|
|
|
syncTidyChecksToTree(config);
|
2018-02-23 11:36:18 +01:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2018-03-26 08:55:23 +02:00
|
|
|
m_tidyChecksWidget->setEnabled(!config.isReadOnly());
|
2018-02-23 11:36:18 +01:00
|
|
|
connectClangTidyItemChanged();
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-16 07:58:53 +02:00
|
|
|
void ClangDiagnosticConfigsWidget::syncTidyChecksToTree(const ClangDiagnosticConfig &config)
|
2018-02-23 11:36:18 +01:00
|
|
|
{
|
2018-05-16 07:58:53 +02:00
|
|
|
m_tidyTreeModel->selectChecks(config.clangTidyChecks());
|
2018-01-26 10:27:58 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangDiagnosticConfigsWidget::syncClazyWidgets(const ClangDiagnosticConfig &config)
|
|
|
|
|
{
|
|
|
|
|
const QString clazyChecks = config.clazyChecks();
|
2018-02-07 09:20:30 +01:00
|
|
|
|
|
|
|
|
QRadioButton *button = m_clazyChecks->clazyRadioDisabled;
|
2018-01-26 10:27:58 +01:00
|
|
|
if (clazyChecks.isEmpty())
|
2018-02-07 09:20:30 +01:00
|
|
|
button = m_clazyChecks->clazyRadioDisabled;
|
|
|
|
|
else if (clazyChecks == "level0")
|
|
|
|
|
button = m_clazyChecks->clazyRadioLevel0;
|
|
|
|
|
else if (clazyChecks == "level1")
|
|
|
|
|
button = m_clazyChecks->clazyRadioLevel1;
|
|
|
|
|
else if (clazyChecks == "level2")
|
|
|
|
|
button = m_clazyChecks->clazyRadioLevel2;
|
|
|
|
|
else if (clazyChecks == "level3")
|
|
|
|
|
button = m_clazyChecks->clazyRadioLevel3;
|
|
|
|
|
|
|
|
|
|
button->setChecked(true);
|
2018-01-26 10:27:58 +01:00
|
|
|
m_clazyChecksWidget->setEnabled(!config.isReadOnly());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangDiagnosticConfigsWidget::updateConfig(const ClangDiagnosticConfig &config)
|
|
|
|
|
{
|
|
|
|
|
m_diagnosticConfigsModel.appendOrUpdate(config);
|
|
|
|
|
emit customConfigsChanged(customConfigs());
|
2016-02-22 17:18:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool ClangDiagnosticConfigsWidget::isConfigChooserEmpty() const
|
|
|
|
|
{
|
2018-05-11 14:53:17 +02:00
|
|
|
return m_ui->configChooserList->count() == 0;
|
2016-02-22 17:18:18 +01:00
|
|
|
}
|
|
|
|
|
|
2016-02-26 17:50:38 +01:00
|
|
|
void ClangDiagnosticConfigsWidget::setDiagnosticOptions(const QString &options)
|
|
|
|
|
{
|
2018-02-06 13:17:36 +01:00
|
|
|
if (options != m_clangBaseChecks->diagnosticOptionsTextEdit->document()->toPlainText()) {
|
2016-02-26 17:50:38 +01:00
|
|
|
disconnectDiagnosticOptionsChanged();
|
2018-02-06 13:17:36 +01:00
|
|
|
m_clangBaseChecks->diagnosticOptionsTextEdit->document()->setPlainText(options);
|
2016-02-26 17:50:38 +01:00
|
|
|
connectDiagnosticOptionsChanged();
|
|
|
|
|
}
|
2018-02-12 11:34:38 +01:00
|
|
|
|
|
|
|
|
const QString errorMessage
|
|
|
|
|
= validateDiagnosticOptions(normalizeDiagnosticInputOptions(options));
|
|
|
|
|
updateValidityWidgets(errorMessage);
|
2016-02-26 17:50:38 +01:00
|
|
|
}
|
|
|
|
|
|
2017-09-08 13:00:44 +02:00
|
|
|
void ClangDiagnosticConfigsWidget::updateValidityWidgets(const QString &errorMessage)
|
|
|
|
|
{
|
|
|
|
|
QString validationResult;
|
|
|
|
|
const Utils::Icon *icon = nullptr;
|
|
|
|
|
QString styleSheet;
|
|
|
|
|
if (errorMessage.isEmpty()) {
|
|
|
|
|
icon = &Utils::Icons::INFO;
|
|
|
|
|
validationResult = tr("Configuration passes sanity checks.");
|
|
|
|
|
} else {
|
|
|
|
|
icon = &Utils::Icons::CRITICAL;
|
|
|
|
|
validationResult = tr("%1").arg(errorMessage);
|
|
|
|
|
styleSheet = "color: red;";
|
|
|
|
|
}
|
|
|
|
|
|
2018-02-12 11:34:38 +01:00
|
|
|
m_ui->infoIcon->setPixmap(icon->pixmap());
|
|
|
|
|
m_ui->infoLabel->setText(validationResult);
|
|
|
|
|
m_ui->infoLabel->setStyleSheet(styleSheet);
|
2017-09-08 13:00:44 +02:00
|
|
|
}
|
|
|
|
|
|
2018-01-26 10:27:58 +01:00
|
|
|
void ClangDiagnosticConfigsWidget::connectClangTidyItemChanged()
|
|
|
|
|
{
|
2018-02-23 11:36:18 +01:00
|
|
|
connect(m_tidyChecks->tidyMode,
|
|
|
|
|
static_cast<void (QComboBox::*)(int)>(&QComboBox::currentIndexChanged),
|
|
|
|
|
this,
|
|
|
|
|
&ClangDiagnosticConfigsWidget::onClangTidyModeChanged);
|
2018-05-16 07:58:53 +02:00
|
|
|
connect(m_tidyTreeModel.get(), &TidyChecksTreeModel::dataChanged,
|
|
|
|
|
this, &ClangDiagnosticConfigsWidget::onClangTidyTreeChanged);
|
2018-01-26 10:27:58 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangDiagnosticConfigsWidget::disconnectClangTidyItemChanged()
|
|
|
|
|
{
|
2018-02-23 11:36:18 +01:00
|
|
|
disconnect(m_tidyChecks->tidyMode,
|
|
|
|
|
static_cast<void (QComboBox::*)(int)>(&QComboBox::currentIndexChanged),
|
|
|
|
|
this,
|
|
|
|
|
&ClangDiagnosticConfigsWidget::onClangTidyModeChanged);
|
2018-05-16 07:58:53 +02:00
|
|
|
disconnect(m_tidyTreeModel.get(), &TidyChecksTreeModel::dataChanged,
|
|
|
|
|
this, &ClangDiagnosticConfigsWidget::onClangTidyTreeChanged);
|
2018-01-26 10:27:58 +01:00
|
|
|
}
|
|
|
|
|
|
2018-02-07 09:20:30 +01:00
|
|
|
void ClangDiagnosticConfigsWidget::connectClazyRadioButtonClicked(QRadioButton *button)
|
|
|
|
|
{
|
|
|
|
|
connect(button,
|
|
|
|
|
&QRadioButton::clicked,
|
|
|
|
|
this,
|
|
|
|
|
&ClangDiagnosticConfigsWidget::onClazyRadioButtonChanged);
|
|
|
|
|
}
|
|
|
|
|
|
2016-02-26 17:50:38 +01:00
|
|
|
void ClangDiagnosticConfigsWidget::connectConfigChooserCurrentIndex()
|
|
|
|
|
{
|
2018-05-14 10:30:09 +02:00
|
|
|
connect(m_ui->configChooserList, &QListWidget::currentRowChanged,
|
|
|
|
|
this, &ClangDiagnosticConfigsWidget::onCurrentConfigChanged);
|
2016-02-26 17:50:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangDiagnosticConfigsWidget::disconnectConfigChooserCurrentIndex()
|
|
|
|
|
{
|
2018-05-14 10:30:09 +02:00
|
|
|
disconnect(m_ui->configChooserList, &QListWidget::currentRowChanged,
|
|
|
|
|
this, &ClangDiagnosticConfigsWidget::onCurrentConfigChanged);
|
2016-02-26 17:50:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangDiagnosticConfigsWidget::connectDiagnosticOptionsChanged()
|
|
|
|
|
{
|
2018-02-06 13:17:36 +01:00
|
|
|
connect(m_clangBaseChecks->diagnosticOptionsTextEdit->document(),
|
|
|
|
|
&QTextDocument::contentsChanged,
|
|
|
|
|
this,
|
|
|
|
|
&ClangDiagnosticConfigsWidget::onDiagnosticOptionsEdited);
|
2016-02-26 17:50:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangDiagnosticConfigsWidget::disconnectDiagnosticOptionsChanged()
|
|
|
|
|
{
|
2018-02-06 13:17:36 +01:00
|
|
|
disconnect(m_clangBaseChecks->diagnosticOptionsTextEdit->document(),
|
|
|
|
|
&QTextDocument::contentsChanged,
|
|
|
|
|
this,
|
|
|
|
|
&ClangDiagnosticConfigsWidget::onDiagnosticOptionsEdited);
|
2016-02-26 17:50:38 +01:00
|
|
|
}
|
|
|
|
|
|
2016-02-22 17:18:18 +01:00
|
|
|
ClangDiagnosticConfigs ClangDiagnosticConfigsWidget::customConfigs() const
|
|
|
|
|
{
|
|
|
|
|
const ClangDiagnosticConfigs allConfigs = m_diagnosticConfigsModel.configs();
|
|
|
|
|
|
|
|
|
|
return Utils::filtered(allConfigs, [](const ClangDiagnosticConfig &config){
|
|
|
|
|
return !config.isReadOnly();
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2018-02-06 13:17:36 +01:00
|
|
|
void ClangDiagnosticConfigsWidget::setupTabs()
|
2018-01-26 10:27:58 +01:00
|
|
|
{
|
2018-02-06 13:17:36 +01:00
|
|
|
m_clangBaseChecks.reset(new CppTools::Ui::ClangBaseChecks);
|
|
|
|
|
m_clangBaseChecksWidget = new QWidget();
|
|
|
|
|
m_clangBaseChecks->setupUi(m_clangBaseChecksWidget);
|
|
|
|
|
|
2018-01-26 10:27:58 +01:00
|
|
|
m_clazyChecks.reset(new CppTools::Ui::ClazyChecks);
|
|
|
|
|
m_clazyChecksWidget = new QWidget();
|
|
|
|
|
m_clazyChecks->setupUi(m_clazyChecksWidget);
|
2018-02-07 09:20:30 +01:00
|
|
|
|
|
|
|
|
connectClazyRadioButtonClicked(m_clazyChecks->clazyRadioDisabled);
|
|
|
|
|
connectClazyRadioButtonClicked(m_clazyChecks->clazyRadioLevel0);
|
|
|
|
|
connectClazyRadioButtonClicked(m_clazyChecks->clazyRadioLevel1);
|
|
|
|
|
connectClazyRadioButtonClicked(m_clazyChecks->clazyRadioLevel2);
|
|
|
|
|
connectClazyRadioButtonClicked(m_clazyChecks->clazyRadioLevel3);
|
2018-01-26 10:27:58 +01:00
|
|
|
|
|
|
|
|
m_tidyChecks.reset(new CppTools::Ui::TidyChecks);
|
|
|
|
|
m_tidyChecksWidget = new QWidget();
|
|
|
|
|
m_tidyChecks->setupUi(m_tidyChecksWidget);
|
2018-05-16 07:58:53 +02:00
|
|
|
m_tidyChecks->checksPrefixesTree->setModel(m_tidyTreeModel.get());
|
|
|
|
|
m_tidyChecks->checksPrefixesTree->expandToDepth(0);
|
2018-05-29 14:38:57 +02:00
|
|
|
m_tidyChecks->checksPrefixesTree->header()->setStretchLastSection(false);
|
|
|
|
|
m_tidyChecks->checksPrefixesTree->header()->setSectionResizeMode(0, QHeaderView::Stretch);
|
2018-05-16 07:58:53 +02:00
|
|
|
connect(m_tidyChecks->plainTextEditButton, &QPushButton::clicked, this, [this]() {
|
|
|
|
|
QDialog dialog;
|
|
|
|
|
dialog.setWindowTitle(tr("Checks"));
|
|
|
|
|
dialog.setLayout(new QVBoxLayout);
|
|
|
|
|
auto *textEdit = new QTextEdit(&dialog);
|
|
|
|
|
dialog.layout()->addWidget(textEdit);
|
|
|
|
|
auto *buttonsBox = new QDialogButtonBox(QDialogButtonBox::Ok | QDialogButtonBox::Cancel);
|
|
|
|
|
dialog.layout()->addWidget(buttonsBox);
|
|
|
|
|
QObject::connect(buttonsBox, &QDialogButtonBox::accepted, &dialog, &QDialog::accept);
|
|
|
|
|
QObject::connect(buttonsBox, &QDialogButtonBox::rejected, &dialog, &QDialog::reject);
|
|
|
|
|
const QString initialChecks = m_tidyTreeModel->selectedChecks();
|
|
|
|
|
textEdit->setPlainText(initialChecks);
|
|
|
|
|
|
|
|
|
|
QObject::connect(&dialog, &QDialog::accepted, [=, &initialChecks]() {
|
|
|
|
|
const QString updatedChecks = textEdit->toPlainText();
|
|
|
|
|
if (updatedChecks == initialChecks)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
disconnectClangTidyItemChanged();
|
|
|
|
|
|
|
|
|
|
// Also throws away invalid options.
|
|
|
|
|
m_tidyTreeModel->selectChecks(updatedChecks);
|
|
|
|
|
onClangTidyTreeChanged();
|
|
|
|
|
|
|
|
|
|
connectClangTidyItemChanged();
|
|
|
|
|
});
|
|
|
|
|
dialog.exec();
|
|
|
|
|
});
|
|
|
|
|
|
2018-01-26 10:27:58 +01:00
|
|
|
connectClangTidyItemChanged();
|
|
|
|
|
|
2018-02-06 13:17:36 +01:00
|
|
|
m_ui->tabWidget->addTab(m_clangBaseChecksWidget, tr("Clang"));
|
|
|
|
|
m_ui->tabWidget->addTab(m_tidyChecksWidget, tr("Clang-Tidy"));
|
|
|
|
|
m_ui->tabWidget->addTab(m_clazyChecksWidget, tr("Clazy"));
|
|
|
|
|
m_ui->tabWidget->setCurrentIndex(0);
|
2018-01-26 10:27:58 +01:00
|
|
|
}
|
|
|
|
|
|
2016-02-22 17:18:18 +01:00
|
|
|
} // CppTools namespace
|
2018-05-16 07:58:53 +02:00
|
|
|
|
|
|
|
|
#include "clangdiagnosticconfigswidget.moc"
|