/**************************************************************************** ** ** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** 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 Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 2.1 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ****************************************************************************/ #include "genericprojectnodes.h" #include "genericproject.h" #include #include #include #include using namespace ProjectExplorer; namespace GenericProjectManager { namespace Internal { GenericProjectNode::GenericProjectNode(GenericProject *project, Core::IDocument *projectFile) : ProjectNode(projectFile->filePath()), m_project(project), m_projectFile(projectFile) { setDisplayName(QFileInfo(projectFile->filePath()).completeBaseName()); } Core::IDocument *GenericProjectNode::projectFile() const { return m_projectFile; } QString GenericProjectNode::projectFilePath() const { return m_projectFile->filePath(); } QHash sortFilesIntoPaths(const QString &base, const QSet files) { QHash filesInPath; const QDir baseDir(base); foreach (const QString &absoluteFileName, files) { QFileInfo fileInfo(absoluteFileName); Utils::FileName absoluteFilePath = Utils::FileName::fromString(fileInfo.path()); QString relativeFilePath; if (absoluteFilePath.isChildOf(baseDir)) { relativeFilePath = absoluteFilePath.relativeChildPath(Utils::FileName::fromString(base)).toString(); } else { // `file' is not part of the project. relativeFilePath = baseDir.relativeFilePath(absoluteFilePath.toString()); if (relativeFilePath.endsWith(QLatin1Char('/'))) relativeFilePath.chop(1); } filesInPath[relativeFilePath].append(absoluteFileName); } return filesInPath; } void GenericProjectNode::refresh(QSet oldFileList) { typedef QHash FilesInPathHash; typedef FilesInPathHash::ConstIterator FilesInPathHashConstIt; if (oldFileList.isEmpty()) { // Only do this once FileNode *projectFilesNode = new FileNode(m_project->filesFileName(), ProjectFileType, /* generated = */ false); FileNode *projectIncludesNode = new FileNode(m_project->includesFileName(), ProjectFileType, /* generated = */ false); FileNode *projectConfigNode = new FileNode(m_project->configFileName(), ProjectFileType, /* generated = */ false); addFileNodes(QList() << projectFilesNode << projectIncludesNode << projectConfigNode, this); } // Do those separately oldFileList.remove(m_project->filesFileName()); oldFileList.remove(m_project->includesFileName()); oldFileList.remove(m_project->configFileName()); QSet newFileList = m_project->files().toSet(); newFileList.remove(m_project->filesFileName()); newFileList.remove(m_project->includesFileName()); newFileList.remove(m_project->configFileName()); QSet removed = oldFileList; removed.subtract(newFileList); QSet added = newFileList; added.subtract(oldFileList); QString baseDir = QFileInfo(path()).absolutePath(); FilesInPathHash filesInPaths = sortFilesIntoPaths(baseDir, added); FilesInPathHashConstIt cend = filesInPaths.constEnd(); for (FilesInPathHashConstIt it = filesInPaths.constBegin(); it != cend; ++it) { const QString &filePath = it.key(); QStringList components; if (!filePath.isEmpty()) components = filePath.split(QLatin1Char('/')); FolderNode *folder = findFolderByName(components, components.size()); if (!folder) folder = createFolderByName(components, components.size()); QList fileNodes; foreach (const QString &file, it.value()) { FileType fileType = SourceType; // ### FIXME FileNode *fileNode = new FileNode(file, fileType, /*generated = */ false); fileNodes.append(fileNode); } addFileNodes(fileNodes, folder); } filesInPaths = sortFilesIntoPaths(baseDir, removed); cend = filesInPaths.constEnd(); for (FilesInPathHashConstIt it = filesInPaths.constBegin(); it != cend; ++it) { const QString &filePath = it.key(); QStringList components; if (!filePath.isEmpty()) components = filePath.split(QLatin1Char('/')); FolderNode *folder = findFolderByName(components, components.size()); QList fileNodes; foreach (const QString &file, it.value()) { foreach (FileNode *fn, folder->fileNodes()) if (fn->path() == file) fileNodes.append(fn); } removeFileNodes(fileNodes, folder); } foreach (FolderNode *fn, subFolderNodes()) removeEmptySubFolders(this, fn); } void GenericProjectNode::removeEmptySubFolders(FolderNode *gparent, FolderNode *parent) { foreach (FolderNode *fn, parent->subFolderNodes()) removeEmptySubFolders(parent, fn); if (parent->subFolderNodes().isEmpty() && parent->fileNodes().isEmpty()) removeFolderNodes(QList() << parent, gparent); } FolderNode *GenericProjectNode::createFolderByName(const QStringList &components, int end) { if (end == 0) return this; QString folderName; for (int i = 0; i < end; ++i) { folderName.append(components.at(i)); folderName += QLatin1Char('/'); } const QString component = components.at(end - 1); const QString baseDir = QFileInfo(path()).path(); FolderNode *folder = new FolderNode(baseDir + QLatin1Char('/') + folderName); folder->setDisplayName(component); FolderNode *parent = findFolderByName(components, end - 1); if (!parent) parent = createFolderByName(components, end - 1); addFolderNodes(QList() << folder, parent); return folder; } FolderNode *GenericProjectNode::findFolderByName(const QStringList &components, int end) { if (end == 0) return this; QString folderName; for (int i = 0; i < end; ++i) { folderName.append(components.at(i)); folderName += QLatin1Char('/'); } FolderNode *parent = findFolderByName(components, end - 1); if (!parent) return 0; const QString baseDir = QFileInfo(path()).path(); foreach (FolderNode *fn, parent->subFolderNodes()) if (fn->path() == baseDir + QLatin1Char('/') + folderName) return fn; return 0; } bool GenericProjectNode::hasBuildTargets() const { return true; } QList GenericProjectNode::supportedActions(Node *node) const { Q_UNUSED(node); return QList() << AddNewFile << AddExistingFile << RemoveFile << Rename; } bool GenericProjectNode::canAddSubProject(const QString &proFilePath) const { Q_UNUSED(proFilePath) return false; } bool GenericProjectNode::addSubProjects(const QStringList &proFilePaths) { Q_UNUSED(proFilePaths) return false; } bool GenericProjectNode::removeSubProjects(const QStringList &proFilePaths) { Q_UNUSED(proFilePaths) return false; } bool GenericProjectNode::addFiles(const QStringList &filePaths, QStringList *notAdded) { Q_UNUSED(notAdded) return m_project->addFiles(filePaths); } bool GenericProjectNode::removeFiles(const QStringList &filePaths, QStringList *notRemoved) { Q_UNUSED(notRemoved) return m_project->removeFiles(filePaths); } bool GenericProjectNode::deleteFiles(const QStringList &filePaths) { Q_UNUSED(filePaths) return false; } bool GenericProjectNode::renameFile(const QString &filePath, const QString &newFilePath) { return m_project->renameFile(filePath, newFilePath); } QList GenericProjectNode::runConfigurationsFor(Node *node) { Q_UNUSED(node) return QList(); } } // namespace Internal } // namespace GenericProjectManager