2022-08-19 15:59:36 +02:00
|
|
|
// Copyright (C) 2020 Alexis Jeandet.
|
2022-12-21 10:12:09 +01:00
|
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
2020-05-01 18:20:56 +02:00
|
|
|
|
|
|
|
|
#include "toolsmodel.h"
|
2020-10-29 10:20:14 +01:00
|
|
|
|
2022-10-10 09:14:04 +02:00
|
|
|
#include "mesonprojectmanagertr.h"
|
2022-10-06 16:58:18 +02:00
|
|
|
#include "mesontools.h"
|
2020-10-29 10:20:14 +01:00
|
|
|
|
2021-01-29 10:01:03 +01:00
|
|
|
#include <projectexplorer/projectexplorerconstants.h>
|
2024-07-24 09:06:49 +02:00
|
|
|
|
2020-10-29 10:20:14 +01:00
|
|
|
#include <utils/qtcassert.h>
|
|
|
|
|
#include <utils/stringutils.h>
|
2024-07-24 09:06:49 +02:00
|
|
|
#include <utils/utilsicons.h>
|
|
|
|
|
|
|
|
|
|
using namespace Utils;
|
|
|
|
|
|
|
|
|
|
namespace MesonProjectManager::Internal {
|
|
|
|
|
|
|
|
|
|
// ToolTreeItem
|
|
|
|
|
|
|
|
|
|
ToolTreeItem::ToolTreeItem(const QString &name)
|
|
|
|
|
: m_name{name}
|
|
|
|
|
, m_id(Id::generate())
|
|
|
|
|
, m_autoDetected{false}
|
|
|
|
|
, m_unsavedChanges{true}
|
|
|
|
|
{
|
|
|
|
|
self_check();
|
|
|
|
|
update_tooltip();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ToolTreeItem::ToolTreeItem(const MesonTools::Tool_t &tool)
|
|
|
|
|
: m_name{tool->name()}
|
|
|
|
|
, m_executable{tool->exe()}
|
|
|
|
|
, m_id{tool->id()}
|
|
|
|
|
, m_autoDetected{tool->autoDetected()}
|
|
|
|
|
{
|
|
|
|
|
m_tooltip = Tr::tr("Version: %1").arg(tool->version().toQString());
|
|
|
|
|
self_check();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ToolTreeItem::ToolTreeItem(const ToolTreeItem &other)
|
|
|
|
|
: m_name{Tr::tr("Clone of %1").arg(other.m_name)}
|
|
|
|
|
, m_executable{other.m_executable}
|
|
|
|
|
, m_id{Id::generate()}
|
|
|
|
|
, m_autoDetected{false}
|
|
|
|
|
, m_unsavedChanges{true}
|
|
|
|
|
{
|
|
|
|
|
self_check();
|
|
|
|
|
update_tooltip();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QVariant ToolTreeItem::data(int column, int role) const
|
|
|
|
|
{
|
|
|
|
|
switch (role) {
|
|
|
|
|
case Qt::DisplayRole: {
|
|
|
|
|
switch (column) {
|
|
|
|
|
case 0: {
|
|
|
|
|
return m_name;
|
|
|
|
|
}
|
|
|
|
|
case 1: {
|
|
|
|
|
return m_executable.toUserOutput();
|
|
|
|
|
}
|
|
|
|
|
} // switch (column)
|
|
|
|
|
return QVariant();
|
|
|
|
|
}
|
|
|
|
|
case Qt::FontRole: {
|
|
|
|
|
QFont font;
|
|
|
|
|
font.setBold(m_unsavedChanges);
|
|
|
|
|
return font;
|
|
|
|
|
}
|
|
|
|
|
case Qt::ToolTipRole: {
|
|
|
|
|
if (!m_pathExists)
|
|
|
|
|
return Tr::tr("Meson executable path does not exist.");
|
|
|
|
|
if (!m_pathIsFile)
|
|
|
|
|
return Tr::tr("Meson executable path is not a file.");
|
|
|
|
|
if (!m_pathIsExecutable)
|
|
|
|
|
return Tr::tr("Meson executable path is not executable.");
|
|
|
|
|
return m_tooltip;
|
|
|
|
|
}
|
|
|
|
|
case Qt::DecorationRole: {
|
|
|
|
|
if (column == 0 && (!m_pathExists || !m_pathIsFile || !m_pathIsExecutable))
|
|
|
|
|
return Icons::CRITICAL.icon();
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ToolTreeItem::update(const QString &name, const FilePath &exe)
|
|
|
|
|
{
|
|
|
|
|
m_unsavedChanges = true;
|
|
|
|
|
m_name = name;
|
|
|
|
|
if (exe != m_executable) {
|
|
|
|
|
m_executable = exe;
|
|
|
|
|
self_check();
|
|
|
|
|
update_tooltip();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ToolTreeItem::self_check()
|
|
|
|
|
{
|
|
|
|
|
m_pathExists = m_executable.exists();
|
|
|
|
|
m_pathIsFile = m_executable.toFileInfo().isFile();
|
|
|
|
|
m_pathIsExecutable = m_executable.toFileInfo().isExecutable();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ToolTreeItem::update_tooltip(const Version &version)
|
|
|
|
|
{
|
|
|
|
|
if (version.isValid)
|
|
|
|
|
m_tooltip = Tr::tr("Version: %1").arg(version.toQString());
|
|
|
|
|
else
|
|
|
|
|
m_tooltip = Tr::tr("Cannot get tool version.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ToolTreeItem::update_tooltip()
|
|
|
|
|
{
|
|
|
|
|
update_tooltip(ToolWrapper::read_version(m_executable));
|
|
|
|
|
}
|
2020-05-01 18:20:56 +02:00
|
|
|
|
2024-07-24 09:06:49 +02:00
|
|
|
// ToolsModel
|
2020-05-01 18:20:56 +02:00
|
|
|
|
|
|
|
|
ToolsModel::ToolsModel()
|
|
|
|
|
{
|
2022-10-10 09:14:04 +02:00
|
|
|
setHeader({Tr::tr("Name"), Tr::tr("Location")});
|
2021-01-29 10:01:03 +01:00
|
|
|
rootItem()->appendChild(
|
2024-07-24 09:06:49 +02:00
|
|
|
new StaticTreeItem({ProjectExplorer::Constants::msgAutoDetected()},
|
|
|
|
|
{ProjectExplorer::Constants::msgAutoDetectedToolTip()}));
|
|
|
|
|
rootItem()->appendChild(new StaticTreeItem(ProjectExplorer::Constants::msgManual()));
|
2022-12-12 09:10:54 +01:00
|
|
|
for (const auto &tool : MesonTools::tools())
|
|
|
|
|
addMesonToolHelper(tool);
|
2020-05-01 18:20:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ToolTreeItem *ToolsModel::mesoneToolTreeItem(const QModelIndex &index) const
|
|
|
|
|
{
|
|
|
|
|
return itemForIndexAtLevel<2>(index);
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-24 09:06:49 +02:00
|
|
|
void ToolsModel::updateItem(const Id &itemId, const QString &name, const FilePath &exe)
|
2020-05-01 18:20:56 +02:00
|
|
|
{
|
|
|
|
|
auto treeItem = findItemAtLevel<2>([itemId](ToolTreeItem *n) { return n->id() == itemId; });
|
|
|
|
|
QTC_ASSERT(treeItem, return );
|
|
|
|
|
treeItem->update(name, exe);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ToolsModel::addMesonTool()
|
|
|
|
|
{
|
2022-10-10 09:14:04 +02:00
|
|
|
manualGroup()->appendChild(new ToolTreeItem{uniqueName(Tr::tr("New Meson or Ninja tool"))});
|
2020-05-01 18:20:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ToolsModel::removeMesonTool(ToolTreeItem *item)
|
|
|
|
|
{
|
|
|
|
|
QTC_ASSERT(item, return );
|
2024-07-24 09:06:49 +02:00
|
|
|
const Id id = item->id();
|
2020-05-01 18:20:56 +02:00
|
|
|
destroyItem(item);
|
2020-06-12 11:18:04 +03:00
|
|
|
m_itemsToRemove.enqueue(id);
|
2020-05-01 18:20:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ToolTreeItem *ToolsModel::cloneMesonTool(ToolTreeItem *item)
|
|
|
|
|
{
|
|
|
|
|
QTC_ASSERT(item, return nullptr);
|
|
|
|
|
auto newItem = new ToolTreeItem(*item);
|
|
|
|
|
manualGroup()->appendChild(newItem);
|
|
|
|
|
return item;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ToolsModel::apply()
|
|
|
|
|
{
|
|
|
|
|
forItemsAtLevel<2>([this](ToolTreeItem *item) {
|
|
|
|
|
if (item->hasUnsavedChanges()) {
|
|
|
|
|
MesonTools::updateTool(item->id(), item->name(), item->executable());
|
|
|
|
|
item->setSaved();
|
2020-10-27 22:36:27 +01:00
|
|
|
emit this->dataChanged(item->index(), item->index());
|
2020-05-01 18:20:56 +02:00
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
while (!m_itemsToRemove.isEmpty()) {
|
|
|
|
|
MesonTools::removeTool(m_itemsToRemove.dequeue());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-12 09:10:54 +01:00
|
|
|
void ToolsModel::addMesonToolHelper(const MesonTools::Tool_t &tool)
|
2020-05-01 18:20:56 +02:00
|
|
|
{
|
|
|
|
|
if (tool->autoDetected())
|
|
|
|
|
autoDetectedGroup()->appendChild(new ToolTreeItem(tool));
|
|
|
|
|
else
|
|
|
|
|
manualGroup()->appendChild(new ToolTreeItem(tool));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString ToolsModel::uniqueName(const QString &baseName)
|
|
|
|
|
{
|
|
|
|
|
QStringList names;
|
2024-07-24 09:06:49 +02:00
|
|
|
forItemsAtLevel<2>([&names](ToolTreeItem *item) { names << item->name(); });
|
2020-05-01 18:20:56 +02:00
|
|
|
return Utils::makeUniquelyNumbered(baseName, names);
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-24 09:06:49 +02:00
|
|
|
TreeItem *ToolsModel::autoDetectedGroup() const
|
2020-05-01 18:20:56 +02:00
|
|
|
{
|
|
|
|
|
return rootItem()->childAt(0);
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-24 09:06:49 +02:00
|
|
|
TreeItem *ToolsModel::manualGroup() const
|
2020-05-01 18:20:56 +02:00
|
|
|
{
|
|
|
|
|
return rootItem()->childAt(1);
|
|
|
|
|
}
|
2020-10-29 10:20:14 +01:00
|
|
|
|
2024-07-24 09:06:49 +02:00
|
|
|
} // MesonProjectManager::Internal
|