forked from qt-creator/qt-creator
Retrieving the list is cheap, there's no reason to store it and pass it through all these functions. Change-Id: I394b512c113af92f60cfb8c2f6561d1617d9cd36 Reviewed-by: hjk <hjk@qt.io>
190 lines
6.6 KiB
C++
190 lines
6.6 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) Filippo Cucchetto <filippocucchetto@gmail.com>
|
|
** Contact: http://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 "nimproject.h"
|
|
#include "nimbuildconfiguration.h"
|
|
#include "nimprojectnode.h"
|
|
#include "nimtoolchain.h"
|
|
|
|
#include "../nimconstants.h"
|
|
|
|
#include <coreplugin/icontext.h>
|
|
#include <coreplugin/progressmanager/progressmanager.h>
|
|
#include <coreplugin/iversioncontrol.h>
|
|
#include <coreplugin/vcsmanager.h>
|
|
#include <projectexplorer/buildconfiguration.h>
|
|
#include <projectexplorer/kit.h>
|
|
#include <projectexplorer/projectexplorerconstants.h>
|
|
#include <projectexplorer/projectnodes.h>
|
|
#include <projectexplorer/target.h>
|
|
#include <projectexplorer/toolchain.h>
|
|
#include <projectexplorer/kitinformation.h>
|
|
#include <texteditor/textdocument.h>
|
|
|
|
#include <utils/algorithm.h>
|
|
#include <utils/qtcassert.h>
|
|
#include <utils/runextensions.h>
|
|
|
|
#include <QFileInfo>
|
|
#include <QQueue>
|
|
|
|
using namespace ProjectExplorer;
|
|
using namespace Utils;
|
|
|
|
namespace Nim {
|
|
|
|
const int MIN_TIME_BETWEEN_PROJECT_SCANS = 4500;
|
|
|
|
NimProject::NimProject(const FileName &fileName) : Project(Constants::C_NIM_MIMETYPE, fileName)
|
|
{
|
|
setId(Constants::C_NIMPROJECT_ID);
|
|
setDisplayName(fileName.toFileInfo().completeBaseName());
|
|
// ensure debugging is enabled (Nim plugin translates nim code to C code)
|
|
setProjectLanguages(Core::Context(ProjectExplorer::Constants::CXX_LANGUAGE_ID));
|
|
|
|
m_projectScanTimer.setSingleShot(true);
|
|
connect(&m_projectScanTimer, &QTimer::timeout, this, &NimProject::collectProjectFiles);
|
|
|
|
connect(&m_futureWatcher, &QFutureWatcher<QList<FileNode *>>::finished, this, &NimProject::updateProject);
|
|
|
|
collectProjectFiles();
|
|
}
|
|
|
|
void NimProject::scheduleProjectScan()
|
|
{
|
|
auto elapsedTime = m_lastProjectScan.elapsed();
|
|
if (elapsedTime < MIN_TIME_BETWEEN_PROJECT_SCANS) {
|
|
if (!m_projectScanTimer.isActive()) {
|
|
m_projectScanTimer.setInterval(MIN_TIME_BETWEEN_PROJECT_SCANS - elapsedTime);
|
|
m_projectScanTimer.start();
|
|
}
|
|
} else {
|
|
collectProjectFiles();
|
|
}
|
|
}
|
|
|
|
bool NimProject::addFiles(const QStringList &filePaths)
|
|
{
|
|
m_excludedFiles = Utils::filtered(m_excludedFiles, [&](const QString &f) { return !filePaths.contains(f); });
|
|
scheduleProjectScan();
|
|
return true;
|
|
}
|
|
|
|
bool NimProject::removeFiles(const QStringList &filePaths)
|
|
{
|
|
m_excludedFiles.append(filePaths);
|
|
m_excludedFiles = Utils::filteredUnique(m_excludedFiles);
|
|
scheduleProjectScan();
|
|
return true;
|
|
}
|
|
|
|
bool NimProject::renameFile(const QString &filePath, const QString &newFilePath)
|
|
{
|
|
Q_UNUSED(filePath)
|
|
m_excludedFiles.removeOne(newFilePath);
|
|
scheduleProjectScan();
|
|
return true;
|
|
}
|
|
|
|
void NimProject::collectProjectFiles()
|
|
{
|
|
m_lastProjectScan.start();
|
|
QTC_ASSERT(!m_futureWatcher.future().isRunning(), return);
|
|
FileName prjDir = projectDirectory();
|
|
QFuture<QList<ProjectExplorer::FileNode *>> future = Utils::runAsync([prjDir] {
|
|
return FileNode::scanForFiles(
|
|
prjDir, [](const FileName &fn) { return new FileNode(fn, FileType::Source, false); });
|
|
});
|
|
m_futureWatcher.setFuture(future);
|
|
Core::ProgressManager::addTask(future, tr("Scanning for Nim files"), "Nim.Project.Scan");
|
|
}
|
|
|
|
void NimProject::updateProject()
|
|
{
|
|
emitParsingStarted();
|
|
const QStringList oldFiles = m_files;
|
|
m_files.clear();
|
|
|
|
std::vector<std::unique_ptr<FileNode>> fileNodes
|
|
= transform<std::vector>(m_futureWatcher.future().result(),
|
|
[](FileNode *fn) { return std::unique_ptr<FileNode>(fn); });
|
|
std::remove_if(std::begin(fileNodes), std::end(fileNodes),
|
|
[this](const std::unique_ptr<FileNode> &fn) {
|
|
const FileName path = fn->filePath();
|
|
const QString fileName = path.fileName();
|
|
return m_excludedFiles.contains(path.toString())
|
|
|| fileName.endsWith(".nimproject", HostOsInfo::fileNameCaseSensitivity())
|
|
|| fileName.contains(".nimproject.user", HostOsInfo::fileNameCaseSensitivity());
|
|
});
|
|
|
|
m_files = transform<QList>(fileNodes, [](const std::unique_ptr<FileNode> &fn) {
|
|
return fn->filePath().toString();
|
|
});
|
|
Utils::sort(m_files);
|
|
|
|
if (oldFiles == m_files)
|
|
return;
|
|
|
|
auto newRoot = std::make_unique<NimProjectNode>(*this, projectDirectory());
|
|
newRoot->setDisplayName(displayName());
|
|
newRoot->addNestedNodes(std::move(fileNodes));
|
|
setRootProjectNode(std::move(newRoot));
|
|
emitParsingFinished(true);
|
|
}
|
|
|
|
QList<Task> NimProject::projectIssues(const Kit *k) const
|
|
{
|
|
QList<Task> result = Project::projectIssues(k);
|
|
auto tc = dynamic_cast<NimToolChain*>(ToolChainKitInformation::toolChain(k, Constants::C_NIMLANGUAGE_ID));
|
|
if (!tc) {
|
|
result.append(createProjectTask(Task::TaskType::Error, tr("No Nim compiler set.")));
|
|
return result;
|
|
}
|
|
if (!tc->compilerCommand().exists())
|
|
result.append(createProjectTask(Task::TaskType::Error, tr("Nim compiler does not exist.")));
|
|
|
|
return result;
|
|
}
|
|
|
|
FileNameList NimProject::nimFiles() const
|
|
{
|
|
return files([](const ProjectExplorer::Node *n) { return AllFiles(n) && n->filePath().endsWith(".nim"); });
|
|
}
|
|
|
|
QVariantMap NimProject::toMap() const
|
|
{
|
|
QVariantMap result = Project::toMap();
|
|
result[Constants::C_NIMPROJECT_EXCLUDEDFILES] = m_excludedFiles;
|
|
return result;
|
|
}
|
|
|
|
Project::RestoreResult NimProject::fromMap(const QVariantMap &map, QString *errorMessage)
|
|
{
|
|
m_excludedFiles = map.value(Constants::C_NIMPROJECT_EXCLUDEDFILES).toStringList();
|
|
return Project::fromMap(map, errorMessage);
|
|
}
|
|
|
|
} // namespace Nim
|