diff options
author | Christian Kandeler <christian.kandeler@qt.io> | 2021-08-30 10:58:08 +0200 |
---|---|---|
committer | Christian Kandeler <christian.kandeler@qt.io> | 2021-09-01 14:53:58 +0000 |
commit | 284817fae6514701902ccdb834c2faa46462f2e8 (patch) | |
tree | 44a8c7d9813dc110b61c4639036366c7696bd7e9 /src/plugins/cppeditor/cpptoolsjsextension.cpp | |
parent | 3e1fa0f170d523971d2c3c12da15a6e291f56511 (diff) | |
download | qt-creator-284817fae6514701902ccdb834c2faa46462f2e8.tar.gz |
Merge CppTools into CppEditor
There was no proper separation of responsibilities between these
plugins. In particular, CppTools had lots of editor-related
functionality, so it's not clear why it was separated out in the first
place.
In fact, for a lot of code, it seemed quite arbitrary where it was put
(just one example: switchHeaderSource() was in CppTools, wheras
switchDeclarationDefinition() was in CppEditor).
Merging the plugins will enable us to get rid of various convoluted
pseudo-abstractions that were only introduced to keep up the artificial
separation.
Change-Id: Iafc3bce625b4794f6d4aa03df6cddc7f2d26716a
Reviewed-by: Christian Stenger <christian.stenger@qt.io>
Diffstat (limited to 'src/plugins/cppeditor/cpptoolsjsextension.cpp')
-rw-r--r-- | src/plugins/cppeditor/cpptoolsjsextension.cpp | 275 |
1 files changed, 275 insertions, 0 deletions
diff --git a/src/plugins/cppeditor/cpptoolsjsextension.cpp b/src/plugins/cppeditor/cpptoolsjsextension.cpp new file mode 100644 index 0000000000..91a747e308 --- /dev/null +++ b/src/plugins/cppeditor/cpptoolsjsextension.cpp @@ -0,0 +1,275 @@ +/**************************************************************************** +** +** 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 "cpptoolsjsextension.h" + +#include "cppfilesettingspage.h" +#include "cpplocatordata.h" +#include "cppworkingcopy.h" + +#include <coreplugin/icore.h> + +#include <projectexplorer/project.h> +#include <projectexplorer/projectnodes.h> +#include <projectexplorer/session.h> + +#include <cplusplus/AST.h> +#include <cplusplus/ASTPath.h> +#include <cplusplus/Overview.h> +#include <utils/codegeneration.h> +#include <utils/fileutils.h> + +#include <QElapsedTimer> +#include <QFileInfo> +#include <QStringList> +#include <QTextStream> + +namespace CppEditor::Internal { + +static QString fileName(const QString &path, const QString &extension) +{ + return Utils::FilePath::fromStringWithExtension(path, extension).toString(); +} + +QString CppToolsJsExtension::headerGuard(const QString &in) const +{ + return Utils::headerGuard(in); +} + +static QStringList parts(const QString &klass) +{ + return klass.split(QStringLiteral("::")); +} + +QStringList CppToolsJsExtension::namespaces(const QString &klass) const +{ + QStringList result = parts(klass); + result.removeLast(); + return result; +} + +bool CppToolsJsExtension::hasNamespaces(const QString &klass) const +{ + return !namespaces(klass).empty(); +} + +QString CppToolsJsExtension::className(const QString &klass) const +{ + QStringList result = parts(klass); + return result.last(); +} + +QString CppToolsJsExtension::classToFileName(const QString &klass, const QString &extension) const +{ + const QString raw = fileName(className(klass), extension); + CppFileSettings settings; + settings.fromSettings(Core::ICore::settings()); + if (!settings.lowerCaseFiles) + return raw; + + QFileInfo fi = QFileInfo(raw); + QString finalPath = fi.path(); + if (finalPath == QStringLiteral(".")) + finalPath.clear(); + if (!finalPath.isEmpty() && !finalPath.endsWith(QLatin1Char('/'))) + finalPath += QLatin1Char('/'); + QString name = fi.baseName().toLower(); + QString ext = fi.completeSuffix(); + if (!ext.isEmpty()) + ext = QString(QLatin1Char('.')) + ext; + return finalPath + name + ext; +} + +QString CppToolsJsExtension::classToHeaderGuard(const QString &klass, const QString &extension) const +{ + return Utils::headerGuard(fileName(className(klass), extension), namespaces(klass)); +} + +QString CppToolsJsExtension::openNamespaces(const QString &klass) const +{ + QString result; + QTextStream str(&result); + Utils::writeOpeningNameSpaces(namespaces(klass), QString(), str); + return result; +} + +QString CppToolsJsExtension::closeNamespaces(const QString &klass) const +{ + QString result; + QTextStream str(&result); + Utils::writeClosingNameSpaces(namespaces(klass), QString(), str); + return result; +} + +bool CppToolsJsExtension::hasQObjectParent(const QString &klassName) const +{ + // This is a synchronous function, but the look-up is potentially expensive. + // Since it's not crucial information, we just abort if retrieving it takes too long, + // in order not to freeze the UI. + // TODO: Any chance to at least cache between successive invocations for the same dialog? + // I don't see it atm... + QElapsedTimer timer; + timer.start(); + static const int timeout = 5000; + + // Find symbol. + QList<IndexItem::Ptr> candidates; + m_locatorData->filterAllFiles([&](const IndexItem::Ptr &item) { + if (timer.elapsed() > timeout) + return IndexItem::VisitorResult::Break; + if (item->scopedSymbolName() == klassName) { + candidates = {item}; + return IndexItem::VisitorResult::Break; + } + if (item->symbolName() == klassName) + candidates << item; + return IndexItem::VisitorResult::Recurse; + }); + if (timer.elapsed() > timeout) + return false; + if (candidates.isEmpty()) + return false; + const IndexItem::Ptr item = candidates.first(); + + // Find class in AST. + const CPlusPlus::Snapshot snapshot = CppModelManager::instance()->snapshot(); + const WorkingCopy workingCopy = CppModelManager::instance()->workingCopy(); + QByteArray source = workingCopy.source(item->fileName()); + if (source.isEmpty()) { + QFile file(item->fileName()); + if (!file.open(QIODevice::ReadOnly | QIODevice::Text)) + return false; + source = file.readAll(); + } + const auto doc = snapshot.preprocessedDocument(source, + Utils::FilePath::fromString(item->fileName())); + if (!doc) + return false; + doc->check(); + if (!doc->translationUnit()) + return false; + if (timer.elapsed() > timeout) + return false; + CPlusPlus::ClassSpecifierAST *classSpec = nullptr; + const QList<CPlusPlus::AST *> astPath = CPlusPlus::ASTPath(doc)(item->line(), item->column()); + for (auto it = astPath.rbegin(); it != astPath.rend(); ++it) { + if ((classSpec = (*it)->asClassSpecifier())) + break; + } + if (!classSpec) + return false; + + // Check whether constructor has a QObject parent parameter. + CPlusPlus::Overview overview; + const CPlusPlus::Class * const klass = classSpec->symbol; + if (!klass) + return false; + for (auto it = klass->memberBegin(); it != klass->memberEnd(); ++it) { + const CPlusPlus::Symbol * const member = *it; + if (overview.prettyName(member->name()) != item->symbolName()) + continue; + const CPlusPlus::Function *function = (*it)->asFunction(); + if (!function) + function = member->type().type()->asFunctionType(); + if (!function) + continue; + for (int i = 0; i < function->argumentCount(); ++i) { + const CPlusPlus::Symbol * const arg = function->argumentAt(i); + const QString argName = overview.prettyName(arg->name()); + const QString argType = overview.prettyType(arg->type()) + .split("::", Qt::SkipEmptyParts).last(); + if (argName == "parent" && argType == "QObject *") + return true; + } + } + + return false; +} + +QString CppToolsJsExtension::includeStatement( + const QString &fullyQualifiedClassName, + const QString &suffix, + const QStringList &specialClasses, + const QString &pathOfIncludingFile + ) +{ + if (fullyQualifiedClassName.isEmpty()) + return {}; + const QString className = parts(fullyQualifiedClassName).constLast(); + if (className.isEmpty() || specialClasses.contains(className)) + return {}; + if (className.startsWith('Q') && className.length() > 2 && className.at(1).isUpper()) + return "#include <" + className + ">\n"; + const auto withUnderScores = [&className] { + QString baseName = className; + baseName[0] = baseName[0].toLower(); + for (int i = 1; i < baseName.length(); ++i) { + if (baseName[i].isUpper()) { + baseName.insert(i, '_'); + baseName[i + 1] = baseName[i + 1].toLower(); + ++i; + } + } + return baseName; + }; + QStringList candidates{className + '.' + suffix}; + bool hasUpperCase = false; + bool hasLowerCase = false; + for (int i = 0; i < className.length() && (!hasUpperCase || !hasLowerCase); ++i) { + if (className.at(i).isUpper()) + hasUpperCase = true; + if (className.at(i).isLower()) + hasLowerCase = true; + } + if (hasUpperCase) + candidates << className.toLower() + '.' + suffix; + if (hasUpperCase && hasLowerCase) + candidates << withUnderScores() + '.' + suffix; + candidates.removeDuplicates(); + using namespace ProjectExplorer; + const auto nodeMatchesFileName = [&candidates](Node *n) { + if (const FileNode * const fileNode = n->asFileNode()) { + if (fileNode->fileType() == FileType::Header + && candidates.contains(fileNode->filePath().fileName())) { + return true; + } + } + return false; + }; + for (const Project * const p : SessionManager::projects()) { + const Node *theNode = p->rootProjectNode()->findNode(nodeMatchesFileName); + if (theNode) { + const bool sameDir = pathOfIncludingFile == theNode->filePath().toFileInfo().path(); + return QString("#include ") + .append(sameDir ? '"' : '<') + .append(theNode->filePath().fileName()) + .append(sameDir ? '"' : '>') + .append('\n'); + } + } + return {}; +} + +} // namespace CppEditor::Internal |