// Copyright (C) Filippo Cucchetto // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0 #include "nimbuildsystem.h" #include "nimconstants.h" #include "nimbleproject.h" #include #include #include #include #include #include #include using namespace ProjectExplorer; using namespace Utils; namespace Nim { const char SETTINGS_KEY[] = "Nim.BuildSystem"; const char EXCLUDED_FILES_KEY[] = "ExcludedFiles"; NimProjectScanner::NimProjectScanner(Project *project) : m_project(project) { m_scanner.setFilter([this](const Utils::MimeType &, const FilePath &fp) { const QString path = fp.toString(); return excludedFiles().contains(path) || path.endsWith(".nimproject") || path.contains(".nimproject.user") || path.contains(".nimble.user"); }); connect(&m_directoryWatcher, &FileSystemWatcher::directoryChanged, this, &NimProjectScanner::directoryChanged); connect(&m_directoryWatcher, &FileSystemWatcher::fileChanged, this, &NimProjectScanner::fileChanged); connect(m_project, &Project::settingsLoaded, this, &NimProjectScanner::loadSettings); connect(m_project, &Project::aboutToSaveSettings, this, &NimProjectScanner::saveSettings); connect(&m_scanner, &TreeScanner::finished, this, [this] { // Collect scanned nodes std::vector> nodes; const TreeScanner::Result scanResult = m_scanner.release(); for (FileNode *node : scanResult.allFiles) { if (!node->path().endsWith(".nim") && !node->path().endsWith(".nimble")) node->setEnabled(false); // Disable files that do not end in .nim nodes.emplace_back(node); } // Sync watched dirs const QSet fsDirs = Utils::transform(nodes, [](const std::unique_ptr &fn) { return fn->directory(); }); const QSet projectDirs = Utils::toSet(m_directoryWatcher.directoryPaths()); m_directoryWatcher.addDirectories(Utils::toList(fsDirs - projectDirs), FileSystemWatcher::WatchAllChanges); m_directoryWatcher.removeDirectories(Utils::toList(projectDirs - fsDirs)); // Sync project files const QSet fsFiles = Utils::transform(nodes, &FileNode::filePath); const QSet projectFiles = Utils::toSet(m_project->files([](const Node *n) { return Project::AllFiles(n); })); if (fsFiles != projectFiles) { auto projectNode = std::make_unique(m_project->projectDirectory()); projectNode->setDisplayName(m_project->displayName()); projectNode->addNestedNodes(std::move(nodes)); m_project->setRootProjectNode(std::move(projectNode)); } emit finished(); }); } void NimProjectScanner::loadSettings() { QVariantMap settings = m_project->namedSettings(SETTINGS_KEY).toMap(); if (settings.contains(EXCLUDED_FILES_KEY)) setExcludedFiles(settings.value(EXCLUDED_FILES_KEY, excludedFiles()).toStringList()); emit requestReparse(); } void NimProjectScanner::saveSettings() { QVariantMap settings; settings.insert(EXCLUDED_FILES_KEY, excludedFiles()); m_project->setNamedSettings(SETTINGS_KEY, settings); } void NimProjectScanner::startScan() { m_scanner.asyncScanForFiles(m_project->projectDirectory()); } void NimProjectScanner::watchProjectFilePath() { m_directoryWatcher.addFile(m_project->projectFilePath(), FileSystemWatcher::WatchModifiedDate); } void NimProjectScanner::setExcludedFiles(const QStringList &list) { static_cast(m_project)->setExcludedFiles(list); } QStringList NimProjectScanner::excludedFiles() const { return static_cast(m_project)->excludedFiles(); } bool NimProjectScanner::addFiles(const QStringList &filePaths) { setExcludedFiles(Utils::filtered(excludedFiles(), [&](const QString & f) { return !filePaths.contains(f); })); emit requestReparse(); return true; } RemovedFilesFromProject NimProjectScanner::removeFiles(const QStringList &filePaths) { setExcludedFiles(Utils::filteredUnique(excludedFiles() + filePaths)); emit requestReparse(); return RemovedFilesFromProject::Ok; } bool NimProjectScanner::renameFile(const QString &, const QString &to) { QStringList files = excludedFiles(); files.removeOne(to); setExcludedFiles(files); emit requestReparse(); return true; } NimBuildSystem::NimBuildSystem(Target *target) : BuildSystem(target), m_projectScanner(target->project()) { connect(&m_projectScanner, &NimProjectScanner::finished, this, [this] { m_guard.markAsSuccess(); m_guard = {}; // Trigger destructor of previous object, emitting parsingFinished() emitBuildSystemUpdated(); }); connect(&m_projectScanner, &NimProjectScanner::requestReparse, this, &NimBuildSystem::requestDelayedParse); connect(&m_projectScanner, &NimProjectScanner::directoryChanged, this, [this] { if (!isWaitingForParse()) requestDelayedParse(); }); requestDelayedParse(); } void NimBuildSystem::triggerParsing() { m_guard = guardParsingRun(); m_projectScanner.startScan(); } FilePath nimPathFromKit(Kit *kit) { auto tc = ToolChainKitAspect::toolChain(kit, Constants::C_NIMLANGUAGE_ID); QTC_ASSERT(tc, return {}); const FilePath command = tc->compilerCommand(); return command.isEmpty() ? FilePath() : command.absolutePath(); } FilePath nimblePathFromKit(Kit *kit) { // There's no extra setting for "nimble", derive it from the "nim" path. const QString nimbleFromPath = QStandardPaths::findExecutable("nimble"); const FilePath nimPath = nimPathFromKit(kit); const FilePath nimbleFromKit = nimPath.pathAppended("nimble").withExecutableSuffix(); return nimbleFromKit.exists() ? nimbleFromKit.canonicalPath() : FilePath::fromString(nimbleFromPath); } bool NimBuildSystem::supportsAction(Node *context, ProjectAction action, const Node *node) const { if (node->asFileNode()) { return action == ProjectAction::Rename || action == ProjectAction::RemoveFile; } if (node->isFolderNodeType() || node->isProjectNodeType()) { return action == ProjectAction::AddNewFile || action == ProjectAction::RemoveFile || action == ProjectAction::AddExistingFile; } return BuildSystem::supportsAction(context, action, node); } bool NimBuildSystem::addFiles(Node *, const FilePaths &filePaths, FilePaths *) { return m_projectScanner.addFiles(Utils::transform(filePaths, &FilePath::toString)); } RemovedFilesFromProject NimBuildSystem::removeFiles(Node *, const FilePaths &filePaths, FilePaths *) { return m_projectScanner.removeFiles(Utils::transform(filePaths, &FilePath::toString)); } bool NimBuildSystem::deleteFiles(Node *, const FilePaths &) { return true; } bool NimBuildSystem::renameFile(Node *, const FilePath &oldFilePath, const FilePath &newFilePath) { return m_projectScanner.renameFile(oldFilePath.toString(), newFilePath.toString()); } } // namespace Nim