2022-08-19 15:59:36 +02:00
|
|
|
// Copyright (C) 2016 The Qt Company Ltd.
|
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
|
2011-07-08 09:56:02 +02:00
|
|
|
|
|
|
|
|
#include "cpptoolsreuse.h"
|
|
|
|
|
|
2022-05-19 14:48:09 +02:00
|
|
|
#include "clangdiagnosticconfigsmodel.h"
|
2021-09-02 08:23:46 +02:00
|
|
|
#include "cppautocompleter.h"
|
2023-08-23 17:26:02 +02:00
|
|
|
#include "cppcanonicalsymbol.h"
|
2016-08-16 13:37:49 +02:00
|
|
|
#include "cppcodemodelsettings.h"
|
2022-06-20 12:57:23 +02:00
|
|
|
#include "cppcompletionassist.h"
|
2021-08-30 10:58:08 +02:00
|
|
|
#include "cppeditorconstants.h"
|
|
|
|
|
#include "cppeditorplugin.h"
|
2023-08-23 17:26:02 +02:00
|
|
|
#include "cppeditorwidget.h"
|
2023-01-11 20:43:10 +01:00
|
|
|
#include "cppeditortr.h"
|
2022-09-21 13:38:42 +02:00
|
|
|
#include "cppfilesettingspage.h"
|
2021-09-02 08:23:46 +02:00
|
|
|
#include "cpphighlighter.h"
|
|
|
|
|
#include "cppqtstyleindenter.h"
|
2022-01-31 13:10:28 +01:00
|
|
|
#include "cppquickfixassistant.h"
|
2021-02-01 15:44:56 +01:00
|
|
|
#include "cpprefactoringchanges.h"
|
2021-05-07 16:10:07 +02:00
|
|
|
#include "projectinfo.h"
|
2015-06-02 17:25:41 +02:00
|
|
|
|
2015-07-10 14:57:42 +02:00
|
|
|
#include <coreplugin/documentmanager.h>
|
2014-05-19 11:17:39 -04:00
|
|
|
#include <coreplugin/editormanager/editormanager.h>
|
2015-02-26 13:38:54 +01:00
|
|
|
#include <coreplugin/idocument.h>
|
2016-08-16 13:37:49 +02:00
|
|
|
#include <coreplugin/messagemanager.h>
|
2023-02-14 15:47:22 +01:00
|
|
|
|
|
|
|
|
#include <projectexplorer/projectmanager.h>
|
|
|
|
|
|
2021-10-13 17:18:07 +02:00
|
|
|
#include <texteditor/codeassist/assistinterface.h>
|
2021-09-02 08:23:46 +02:00
|
|
|
#include <texteditor/textdocument.h>
|
2014-05-19 11:17:39 -04:00
|
|
|
|
2021-10-13 17:18:07 +02:00
|
|
|
#include <cplusplus/BackwardsScanner.h>
|
2023-08-23 17:26:02 +02:00
|
|
|
#include <cplusplus/declarationcomments.h>
|
2011-11-14 13:14:39 +01:00
|
|
|
#include <cplusplus/LookupContext.h>
|
2021-10-13 17:18:07 +02:00
|
|
|
#include <cplusplus/Overview.h>
|
|
|
|
|
#include <cplusplus/SimpleLexer.h>
|
2023-02-14 15:47:22 +01:00
|
|
|
|
2015-11-03 13:52:52 +01:00
|
|
|
#include <utils/algorithm.h>
|
2014-05-19 11:17:39 -04:00
|
|
|
#include <utils/qtcassert.h>
|
2023-08-23 17:26:02 +02:00
|
|
|
#include <utils/textfileformat.h>
|
|
|
|
|
#include <utils/textutils.h>
|
2011-11-14 13:14:39 +01:00
|
|
|
|
2015-05-12 14:20:32 +02:00
|
|
|
#include <QDebug>
|
2023-08-23 17:26:02 +02:00
|
|
|
#include <QElapsedTimer>
|
|
|
|
|
#include <QHash>
|
2019-02-08 16:11:27 +01:00
|
|
|
#include <QRegularExpression>
|
2012-02-15 10:42:41 +01:00
|
|
|
#include <QSet>
|
2023-08-23 17:26:02 +02:00
|
|
|
#include <QStringView>
|
2014-06-10 16:05:14 -04:00
|
|
|
#include <QTextCursor>
|
|
|
|
|
#include <QTextDocument>
|
2014-05-19 11:17:39 -04:00
|
|
|
|
2023-09-22 15:20:32 +02:00
|
|
|
#include <optional>
|
2023-08-23 17:26:02 +02:00
|
|
|
#include <vector>
|
|
|
|
|
|
2011-11-14 13:14:39 +01:00
|
|
|
using namespace CPlusPlus;
|
2022-11-24 14:11:12 +01:00
|
|
|
using namespace Utils;
|
2011-11-14 13:14:39 +01:00
|
|
|
|
2021-08-30 10:58:08 +02:00
|
|
|
namespace CppEditor {
|
2011-07-08 09:56:02 +02:00
|
|
|
|
2019-02-08 16:11:27 +01:00
|
|
|
static int skipChars(QTextCursor *tc,
|
|
|
|
|
QTextCursor::MoveOperation op,
|
|
|
|
|
int offset,
|
|
|
|
|
std::function<bool(const QChar &)> skip)
|
2012-02-16 15:09:56 +01:00
|
|
|
{
|
2019-02-08 16:11:27 +01:00
|
|
|
const QTextDocument *doc = tc->document();
|
2011-07-08 09:56:02 +02:00
|
|
|
if (!doc)
|
2019-02-08 16:11:27 +01:00
|
|
|
return 0;
|
|
|
|
|
QChar ch = doc->characterAt(tc->position() + offset);
|
|
|
|
|
if (ch.isNull())
|
|
|
|
|
return 0;
|
|
|
|
|
int count = 0;
|
|
|
|
|
while (skip(ch)) {
|
|
|
|
|
if (tc->movePosition(op))
|
|
|
|
|
++count;
|
|
|
|
|
else
|
|
|
|
|
break;
|
|
|
|
|
ch = doc->characterAt(tc->position() + offset);
|
2011-07-08 09:56:02 +02:00
|
|
|
}
|
2019-02-08 16:11:27 +01:00
|
|
|
return count;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int skipCharsForward(QTextCursor *tc, std::function<bool(const QChar &)> skip)
|
|
|
|
|
{
|
|
|
|
|
return skipChars(tc, QTextCursor::NextCharacter, 0, skip);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int skipCharsBackward(QTextCursor *tc, std::function<bool(const QChar &)> skip)
|
|
|
|
|
{
|
|
|
|
|
return skipChars(tc, QTextCursor::PreviousCharacter, -1, skip);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QStringList identifierWordsUnderCursor(const QTextCursor &tc)
|
|
|
|
|
{
|
|
|
|
|
const QTextDocument *document = tc.document();
|
|
|
|
|
if (!document)
|
|
|
|
|
return {};
|
|
|
|
|
const auto isSpace = [](const QChar &c) { return c.isSpace(); };
|
|
|
|
|
const auto isColon = [](const QChar &c) { return c == ':'; };
|
|
|
|
|
const auto isValidIdentifierCharAt = [document](const QTextCursor &tc) {
|
|
|
|
|
return isValidIdentifierChar(document->characterAt(tc.position()));
|
|
|
|
|
};
|
|
|
|
|
// move to the end
|
|
|
|
|
QTextCursor endCursor(tc);
|
|
|
|
|
do {
|
|
|
|
|
moveCursorToEndOfIdentifier(&endCursor);
|
|
|
|
|
// possibly skip ::
|
|
|
|
|
QTextCursor temp(endCursor);
|
|
|
|
|
skipCharsForward(&temp, isSpace);
|
|
|
|
|
const int colons = skipCharsForward(&temp, isColon);
|
|
|
|
|
skipCharsForward(&temp, isSpace);
|
|
|
|
|
if (colons == 2 && isValidIdentifierCharAt(temp))
|
|
|
|
|
endCursor = temp;
|
|
|
|
|
} while (isValidIdentifierCharAt(endCursor));
|
|
|
|
|
|
|
|
|
|
QStringList results;
|
|
|
|
|
QTextCursor startCursor(endCursor);
|
|
|
|
|
do {
|
|
|
|
|
moveCursorToStartOfIdentifier(&startCursor);
|
|
|
|
|
if (startCursor.position() == endCursor.position())
|
|
|
|
|
break;
|
|
|
|
|
QTextCursor temp(endCursor);
|
|
|
|
|
temp.setPosition(startCursor.position(), QTextCursor::KeepAnchor);
|
|
|
|
|
results.append(temp.selectedText().remove(QRegularExpression("\\s")));
|
|
|
|
|
// possibly skip ::
|
|
|
|
|
temp = startCursor;
|
|
|
|
|
skipCharsBackward(&temp, isSpace);
|
|
|
|
|
const int colons = skipCharsBackward(&temp, isColon);
|
|
|
|
|
skipCharsBackward(&temp, isSpace);
|
|
|
|
|
if (colons == 2
|
|
|
|
|
&& isValidIdentifierChar(document->characterAt(temp.position() - 1))) {
|
|
|
|
|
startCursor = temp;
|
|
|
|
|
}
|
|
|
|
|
} while (!isValidIdentifierCharAt(startCursor));
|
|
|
|
|
return results;
|
2011-07-08 09:56:02 +02:00
|
|
|
}
|
|
|
|
|
|
2012-02-16 15:09:56 +01:00
|
|
|
void moveCursorToEndOfIdentifier(QTextCursor *tc)
|
|
|
|
|
{
|
2019-02-08 16:11:27 +01:00
|
|
|
skipCharsForward(tc, isValidIdentifierChar);
|
2012-02-16 15:09:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void moveCursorToStartOfIdentifier(QTextCursor *tc)
|
|
|
|
|
{
|
2019-02-08 16:11:27 +01:00
|
|
|
skipCharsBackward(tc, isValidIdentifierChar);
|
2012-02-16 15:09:56 +01:00
|
|
|
}
|
|
|
|
|
|
2011-11-14 13:14:39 +01:00
|
|
|
static bool isOwnershipRAIIName(const QString &name)
|
|
|
|
|
{
|
|
|
|
|
static QSet<QString> knownNames;
|
|
|
|
|
if (knownNames.isEmpty()) {
|
|
|
|
|
// Qt
|
|
|
|
|
knownNames.insert(QLatin1String("QScopedPointer"));
|
|
|
|
|
knownNames.insert(QLatin1String("QScopedArrayPointer"));
|
|
|
|
|
knownNames.insert(QLatin1String("QMutexLocker"));
|
|
|
|
|
knownNames.insert(QLatin1String("QReadLocker"));
|
|
|
|
|
knownNames.insert(QLatin1String("QWriteLocker"));
|
|
|
|
|
// Standard C++
|
|
|
|
|
knownNames.insert(QLatin1String("auto_ptr"));
|
|
|
|
|
knownNames.insert(QLatin1String("unique_ptr"));
|
|
|
|
|
// Boost
|
|
|
|
|
knownNames.insert(QLatin1String("scoped_ptr"));
|
|
|
|
|
knownNames.insert(QLatin1String("scoped_array"));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return knownNames.contains(name);
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-04 17:01:07 +02:00
|
|
|
bool isOwnershipRAIIType(Symbol *symbol, const LookupContext &context)
|
2011-11-14 13:14:39 +01:00
|
|
|
{
|
|
|
|
|
if (!symbol)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
// This is not a "real" comparison of types. What we do is to resolve the symbol
|
|
|
|
|
// in question and then try to match its name with already known ones.
|
2022-06-23 16:56:36 +02:00
|
|
|
if (symbol->asDeclaration()) {
|
2011-11-14 13:14:39 +01:00
|
|
|
Declaration *declaration = symbol->asDeclaration();
|
|
|
|
|
const NamedType *namedType = declaration->type()->asNamedType();
|
|
|
|
|
if (namedType) {
|
2015-11-19 13:49:26 +01:00
|
|
|
ClassOrNamespace *clazz = context.lookupType(namedType->name(),
|
2011-11-14 13:14:39 +01:00
|
|
|
declaration->enclosingScope());
|
|
|
|
|
if (clazz && !clazz->symbols().isEmpty()) {
|
|
|
|
|
Overview overview;
|
|
|
|
|
Symbol *symbol = clazz->symbols().at(0);
|
|
|
|
|
return isOwnershipRAIIName(overview.prettyName(symbol->name()));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-08 09:48:27 -04:00
|
|
|
bool isValidAsciiIdentifierChar(const QChar &ch)
|
|
|
|
|
{
|
2014-09-22 10:30:39 +02:00
|
|
|
return ch.isLetterOrNumber() || ch == QLatin1Char('_');
|
2014-05-08 09:48:27 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool isValidFirstIdentifierChar(const QChar &ch)
|
|
|
|
|
{
|
|
|
|
|
return ch.isLetter() || ch == QLatin1Char('_') || ch.isHighSurrogate() || ch.isLowSurrogate();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool isValidIdentifierChar(const QChar &ch)
|
|
|
|
|
{
|
|
|
|
|
return isValidFirstIdentifierChar(ch) || ch.isNumber();
|
|
|
|
|
}
|
|
|
|
|
|
2011-12-21 18:24:14 +01:00
|
|
|
bool isValidIdentifier(const QString &s)
|
|
|
|
|
{
|
|
|
|
|
const int length = s.length();
|
|
|
|
|
for (int i = 0; i < length; ++i) {
|
|
|
|
|
const QChar &c = s.at(i);
|
|
|
|
|
if (i == 0) {
|
2014-05-08 09:48:27 -04:00
|
|
|
if (!isValidFirstIdentifierChar(c))
|
2011-12-21 18:24:14 +01:00
|
|
|
return false;
|
|
|
|
|
} else {
|
2014-05-08 09:48:27 -04:00
|
|
|
if (!isValidIdentifierChar(c))
|
2011-12-21 18:24:14 +01:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-15 17:39:32 +02:00
|
|
|
bool isQtKeyword(QStringView text)
|
2010-11-03 11:02:25 +01:00
|
|
|
{
|
|
|
|
|
switch (text.length()) {
|
|
|
|
|
case 4:
|
|
|
|
|
switch (text.at(0).toLatin1()) {
|
|
|
|
|
case 'e':
|
|
|
|
|
if (text == QLatin1String("emit"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
case 'S':
|
|
|
|
|
if (text == QLatin1String("SLOT"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 5:
|
|
|
|
|
if (text.at(0) == QLatin1Char('s') && text == QLatin1String("slots"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 6:
|
|
|
|
|
if (text.at(0) == QLatin1Char('S') && text == QLatin1String("SIGNAL"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 7:
|
|
|
|
|
switch (text.at(0).toLatin1()) {
|
|
|
|
|
case 's':
|
|
|
|
|
if (text == QLatin1String("signals"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
case 'f':
|
|
|
|
|
if (text == QLatin1String("foreach") || text == QLatin1String("forever"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2011-11-14 13:14:39 +01:00
|
|
|
|
2014-06-10 16:05:14 -04:00
|
|
|
QString identifierUnderCursor(QTextCursor *cursor)
|
|
|
|
|
{
|
|
|
|
|
cursor->movePosition(QTextCursor::StartOfWord);
|
|
|
|
|
cursor->movePosition(QTextCursor::EndOfWord, QTextCursor::KeepAnchor);
|
|
|
|
|
return cursor->selectedText();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const Macro *findCanonicalMacro(const QTextCursor &cursor, Document::Ptr document)
|
|
|
|
|
{
|
2019-01-14 01:40:53 +01:00
|
|
|
QTC_ASSERT(document, return nullptr);
|
2014-06-10 16:05:14 -04:00
|
|
|
|
2023-05-09 15:54:09 +02:00
|
|
|
if (const Macro *macro = document->findMacroDefinitionAt(cursor.blockNumber() + 1)) {
|
2014-06-10 16:05:14 -04:00
|
|
|
QTextCursor macroCursor = cursor;
|
2021-08-30 10:58:08 +02:00
|
|
|
const QByteArray name = identifierUnderCursor(¯oCursor).toUtf8();
|
2014-06-10 16:05:14 -04:00
|
|
|
if (macro->name() == name)
|
|
|
|
|
return macro;
|
|
|
|
|
} else if (const Document::MacroUse *use = document->findMacroUseAt(cursor.position())) {
|
|
|
|
|
return &use->macro();
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-14 01:40:53 +01:00
|
|
|
return nullptr;
|
2014-06-10 16:05:14 -04:00
|
|
|
}
|
|
|
|
|
|
2021-10-13 17:18:07 +02:00
|
|
|
bool isInCommentOrString(const TextEditor::AssistInterface *interface,
|
|
|
|
|
CPlusPlus::LanguageFeatures features)
|
|
|
|
|
{
|
|
|
|
|
QTextCursor tc(interface->textDocument());
|
|
|
|
|
tc.setPosition(interface->position());
|
2023-10-17 10:34:57 +02:00
|
|
|
return isInCommentOrString(tc, features);
|
|
|
|
|
}
|
2021-10-13 17:18:07 +02:00
|
|
|
|
2023-10-17 10:34:57 +02:00
|
|
|
bool isInCommentOrString(const QTextCursor &cursor, CPlusPlus::LanguageFeatures features)
|
|
|
|
|
{
|
2021-10-13 17:18:07 +02:00
|
|
|
SimpleLexer tokenize;
|
|
|
|
|
features.qtMocRunEnabled = true;
|
|
|
|
|
tokenize.setLanguageFeatures(features);
|
|
|
|
|
tokenize.setSkipComments(false);
|
2023-10-17 10:34:57 +02:00
|
|
|
const Tokens &tokens = tokenize(cursor.block().text(),
|
|
|
|
|
BackwardsScanner::previousBlockState(cursor.block()));
|
|
|
|
|
const int tokenIdx = SimpleLexer::tokenBefore(tokens, qMax(0, cursor.positionInBlock() - 1));
|
2021-10-13 17:18:07 +02:00
|
|
|
const Token tk = (tokenIdx == -1) ? Token() : tokens.at(tokenIdx);
|
|
|
|
|
|
|
|
|
|
if (tk.isComment())
|
|
|
|
|
return true;
|
2023-10-17 10:34:57 +02:00
|
|
|
if (!tk.isStringLiteral())
|
2021-10-13 17:18:07 +02:00
|
|
|
return false;
|
|
|
|
|
if (tokens.size() == 3 && tokens.at(0).kind() == T_POUND
|
2023-10-17 10:34:57 +02:00
|
|
|
&& tokens.at(1).kind() == T_IDENTIFIER) {
|
|
|
|
|
const QString &line = cursor.block().text();
|
2021-10-13 17:18:07 +02:00
|
|
|
const Token &idToken = tokens.at(1);
|
2022-07-13 10:06:41 +02:00
|
|
|
QStringView identifier = QStringView(line).mid(idToken.utf16charsBegin(),
|
|
|
|
|
idToken.utf16chars());
|
2021-10-13 17:18:07 +02:00
|
|
|
if (identifier == QLatin1String("include")
|
2023-10-17 10:34:57 +02:00
|
|
|
|| identifier == QLatin1String("include_next")
|
|
|
|
|
|| (features.objCEnabled && identifier == QLatin1String("import"))) {
|
2021-10-13 17:18:07 +02:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2022-01-31 13:10:28 +01:00
|
|
|
TextEditor::QuickFixOperations quickFixOperations(const TextEditor::AssistInterface *interface)
|
|
|
|
|
{
|
|
|
|
|
return Internal::quickFixOperations(interface);
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-20 12:57:23 +02:00
|
|
|
CppCompletionAssistProcessor *getCppCompletionAssistProcessor()
|
|
|
|
|
{
|
|
|
|
|
return new Internal::InternalCppCompletionAssistProcessor();
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-03 11:13:27 +01:00
|
|
|
CppCodeModelSettings *codeModelSettings()
|
2015-06-02 17:25:41 +02:00
|
|
|
{
|
2021-08-30 10:58:08 +02:00
|
|
|
return Internal::CppEditorPlugin::instance()->codeModelSettings();
|
2015-06-02 17:25:41 +02:00
|
|
|
}
|
|
|
|
|
|
2016-08-16 13:37:49 +02:00
|
|
|
int indexerFileSizeLimitInMb()
|
2015-05-12 14:20:32 +02:00
|
|
|
{
|
2020-02-03 11:13:27 +01:00
|
|
|
const CppCodeModelSettings *settings = codeModelSettings();
|
2016-08-16 13:37:49 +02:00
|
|
|
QTC_ASSERT(settings, return -1);
|
2015-05-12 14:20:32 +02:00
|
|
|
|
2016-08-16 13:37:49 +02:00
|
|
|
if (settings->skipIndexingBigFiles())
|
|
|
|
|
return settings->indexerFileSizeLimitInMb();
|
2015-05-12 14:20:32 +02:00
|
|
|
|
2016-08-16 13:37:49 +02:00
|
|
|
return -1;
|
2015-05-12 14:20:32 +02:00
|
|
|
}
|
|
|
|
|
|
2022-11-24 14:11:12 +01:00
|
|
|
bool fileSizeExceedsLimit(const FilePath &filePath, int sizeLimitInMb)
|
2015-05-12 14:20:32 +02:00
|
|
|
{
|
2016-08-16 13:37:49 +02:00
|
|
|
if (sizeLimitInMb <= 0)
|
2015-05-12 14:20:32 +02:00
|
|
|
return false;
|
|
|
|
|
|
2022-11-24 14:11:12 +01:00
|
|
|
const qint64 fileSizeInMB = filePath.fileSize() / (1000 * 1000);
|
2016-08-16 13:37:49 +02:00
|
|
|
if (fileSizeInMB > sizeLimitInMb) {
|
2023-01-11 20:43:10 +01:00
|
|
|
const QString msg = Tr::tr("C++ Indexer: Skipping file \"%1\" because it is too big.")
|
2022-11-24 14:11:12 +01:00
|
|
|
.arg(filePath.displayName());
|
2018-11-15 15:08:44 +01:00
|
|
|
|
2020-12-15 10:13:13 +01:00
|
|
|
QMetaObject::invokeMethod(Core::MessageManager::instance(),
|
|
|
|
|
[msg]() { Core::MessageManager::writeSilently(msg); });
|
2018-11-15 15:08:44 +01:00
|
|
|
|
2015-05-12 14:20:32 +02:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-30 10:34:05 +01:00
|
|
|
UsePrecompiledHeaders getPchUsage()
|
2016-11-02 12:21:54 +03:00
|
|
|
{
|
2020-02-03 11:13:27 +01:00
|
|
|
const CppCodeModelSettings *cms = codeModelSettings();
|
2016-11-02 12:21:54 +03:00
|
|
|
if (cms->pchUsage() == CppCodeModelSettings::PchUse_None)
|
2018-11-30 10:34:05 +01:00
|
|
|
return UsePrecompiledHeaders::No;
|
|
|
|
|
return UsePrecompiledHeaders::Yes;
|
2016-11-02 12:21:54 +03:00
|
|
|
}
|
|
|
|
|
|
2019-09-25 15:46:15 +02:00
|
|
|
static void addBuiltinConfigs(ClangDiagnosticConfigsModel &model)
|
|
|
|
|
{
|
|
|
|
|
ClangDiagnosticConfig config;
|
|
|
|
|
|
|
|
|
|
// Questionable constructs
|
|
|
|
|
config = ClangDiagnosticConfig();
|
2019-09-27 13:50:40 +02:00
|
|
|
config.setId(Constants::CPP_CLANG_DIAG_CONFIG_QUESTIONABLE);
|
2023-01-11 20:43:10 +01:00
|
|
|
config.setDisplayName(Tr::tr("Checks for questionable constructs"));
|
2019-09-25 15:46:15 +02:00
|
|
|
config.setIsReadOnly(true);
|
2019-09-27 13:50:40 +02:00
|
|
|
config.setClangOptions({
|
|
|
|
|
"-Wall",
|
|
|
|
|
"-Wextra",
|
2019-09-25 15:46:15 +02:00
|
|
|
});
|
2020-05-27 15:04:59 +02:00
|
|
|
config.setClazyMode(ClangDiagnosticConfig::ClazyMode::UseCustomChecks);
|
|
|
|
|
config.setClangTidyMode(ClangDiagnosticConfig::TidyMode::UseCustomChecks);
|
2019-09-25 15:46:15 +02:00
|
|
|
model.appendOrUpdate(config);
|
|
|
|
|
|
2019-09-27 13:50:40 +02:00
|
|
|
// Warning flags from build system
|
2019-09-25 15:46:15 +02:00
|
|
|
config = ClangDiagnosticConfig();
|
2021-02-18 14:38:17 +01:00
|
|
|
config.setId(Constants::CPP_CLANG_DIAG_CONFIG_BUILDSYSTEM);
|
2023-01-11 20:43:10 +01:00
|
|
|
config.setDisplayName(Tr::tr("Build-system warnings"));
|
2019-09-25 15:46:15 +02:00
|
|
|
config.setIsReadOnly(true);
|
2020-05-27 15:04:59 +02:00
|
|
|
config.setClazyMode(ClangDiagnosticConfig::ClazyMode::UseCustomChecks);
|
|
|
|
|
config.setClangTidyMode(ClangDiagnosticConfig::TidyMode::UseCustomChecks);
|
2019-09-25 15:46:15 +02:00
|
|
|
config.setUseBuildSystemWarnings(true);
|
|
|
|
|
model.appendOrUpdate(config);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ClangDiagnosticConfigsModel diagnosticConfigsModel(const ClangDiagnosticConfigs &customConfigs)
|
|
|
|
|
{
|
|
|
|
|
ClangDiagnosticConfigsModel model;
|
|
|
|
|
addBuiltinConfigs(model);
|
|
|
|
|
for (const ClangDiagnosticConfig &config : customConfigs)
|
|
|
|
|
model.appendOrUpdate(config);
|
|
|
|
|
return model;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ClangDiagnosticConfigsModel diagnosticConfigsModel()
|
|
|
|
|
{
|
2022-05-19 14:48:09 +02:00
|
|
|
return diagnosticConfigsModel(ClangdSettings::instance().customDiagnosticConfigs());
|
2019-09-25 15:46:15 +02:00
|
|
|
}
|
|
|
|
|
|
2021-02-01 15:44:56 +01:00
|
|
|
NSVisitor::NSVisitor(const CppRefactoringFile *file, const QStringList &namespaces, int symbolPos)
|
|
|
|
|
: ASTVisitor(file->cppDocument()->translationUnit()),
|
|
|
|
|
m_file(file),
|
|
|
|
|
m_remainingNamespaces(namespaces),
|
|
|
|
|
m_symbolPos(symbolPos)
|
|
|
|
|
{}
|
|
|
|
|
|
2021-08-30 10:58:08 +02:00
|
|
|
bool NSVisitor::preVisit(AST *ast)
|
2021-02-01 15:44:56 +01:00
|
|
|
{
|
|
|
|
|
if (!m_firstToken)
|
|
|
|
|
m_firstToken = ast;
|
|
|
|
|
if (m_file->startOf(ast) >= m_symbolPos)
|
|
|
|
|
m_done = true;
|
|
|
|
|
return !m_done;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool NSVisitor::visit(NamespaceAST *ns)
|
|
|
|
|
{
|
|
|
|
|
if (!m_firstNamespace)
|
|
|
|
|
m_firstNamespace = ns;
|
|
|
|
|
if (m_remainingNamespaces.isEmpty()) {
|
|
|
|
|
m_done = true;
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString name;
|
|
|
|
|
const Identifier * const id = translationUnit()->identifier(ns->identifier_token);
|
|
|
|
|
if (id)
|
|
|
|
|
name = QString::fromUtf8(id->chars(), id->size());
|
|
|
|
|
if (name != m_remainingNamespaces.first())
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
if (!ns->linkage_body) {
|
|
|
|
|
m_done = true;
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m_enclosingNamespace = ns;
|
|
|
|
|
m_remainingNamespaces.removeFirst();
|
|
|
|
|
return !m_remainingNamespaces.isEmpty();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void NSVisitor::postVisit(AST *ast)
|
|
|
|
|
{
|
|
|
|
|
if (ast == m_enclosingNamespace)
|
|
|
|
|
m_done = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief The NSCheckerVisitor class checks which namespaces are missing for a given list
|
|
|
|
|
* of enclosing namespaces at a given position
|
|
|
|
|
*/
|
|
|
|
|
NSCheckerVisitor::NSCheckerVisitor(const CppRefactoringFile *file, const QStringList &namespaces,
|
|
|
|
|
int symbolPos)
|
|
|
|
|
: ASTVisitor(file->cppDocument()->translationUnit())
|
|
|
|
|
, m_file(file)
|
|
|
|
|
, m_remainingNamespaces(namespaces)
|
|
|
|
|
, m_symbolPos(symbolPos)
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
bool NSCheckerVisitor::preVisit(AST *ast)
|
|
|
|
|
{
|
|
|
|
|
if (m_file->startOf(ast) >= m_symbolPos)
|
|
|
|
|
m_done = true;
|
|
|
|
|
return !m_done;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void NSCheckerVisitor::postVisit(AST *ast)
|
|
|
|
|
{
|
|
|
|
|
if (!m_done && m_file->endOf(ast) > m_symbolPos)
|
|
|
|
|
m_done = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool NSCheckerVisitor::visit(NamespaceAST *ns)
|
|
|
|
|
{
|
|
|
|
|
if (m_remainingNamespaces.isEmpty())
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
QString name = getName(ns);
|
|
|
|
|
if (name != m_remainingNamespaces.first())
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
m_enteredNamespaces.push_back(ns);
|
|
|
|
|
m_remainingNamespaces.removeFirst();
|
|
|
|
|
// if we reached the searched namespace we don't have to search deeper
|
|
|
|
|
return !m_remainingNamespaces.isEmpty();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool NSCheckerVisitor::visit(UsingDirectiveAST *usingNS)
|
|
|
|
|
{
|
|
|
|
|
// example: we search foo::bar and get 'using namespace foo;using namespace foo::bar;'
|
|
|
|
|
const QString fullName = Overview{}.prettyName(usingNS->name->name);
|
|
|
|
|
const QStringList namespaces = fullName.split("::");
|
|
|
|
|
if (namespaces.length() > m_remainingNamespaces.length())
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
// from other using namespace statements
|
|
|
|
|
const auto curList = m_usingsPerNamespace.find(currentNamespace());
|
|
|
|
|
const bool isCurListValid = curList != m_usingsPerNamespace.end();
|
|
|
|
|
|
|
|
|
|
const bool startEqual = std::equal(namespaces.cbegin(),
|
|
|
|
|
namespaces.cend(),
|
|
|
|
|
m_remainingNamespaces.cbegin());
|
|
|
|
|
if (startEqual) {
|
|
|
|
|
if (isCurListValid) {
|
|
|
|
|
if (namespaces.length() > curList->second.length()) {
|
|
|
|
|
// eg. we already have 'using namespace foo;' and
|
|
|
|
|
// now get 'using namespace foo::bar;'
|
|
|
|
|
curList->second = namespaces;
|
|
|
|
|
}
|
|
|
|
|
// the other case: first 'using namespace foo::bar;' and now 'using namespace foo;'
|
|
|
|
|
} else
|
|
|
|
|
m_usingsPerNamespace.emplace(currentNamespace(), namespaces);
|
|
|
|
|
} else if (isCurListValid) {
|
|
|
|
|
// ex: we have already 'using namespace foo;' and get 'using namespace bar;' now
|
|
|
|
|
QStringList newlist = curList->second;
|
|
|
|
|
newlist.append(namespaces);
|
|
|
|
|
if (newlist.length() <= m_remainingNamespaces.length()) {
|
|
|
|
|
const bool startEqual = std::equal(newlist.cbegin(),
|
|
|
|
|
newlist.cend(),
|
|
|
|
|
m_remainingNamespaces.cbegin());
|
|
|
|
|
if (startEqual)
|
|
|
|
|
curList->second.append(namespaces);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void NSCheckerVisitor::endVisit(NamespaceAST *ns)
|
|
|
|
|
{
|
|
|
|
|
// if the symbolPos was in the namespace and the
|
|
|
|
|
// namespace has no children, m_done should be true
|
|
|
|
|
postVisit(ns);
|
|
|
|
|
if (!m_done && currentNamespace() == ns) {
|
|
|
|
|
// we were not succesfull in this namespace, so undo all changes
|
|
|
|
|
m_remainingNamespaces.push_front(getName(currentNamespace()));
|
|
|
|
|
m_usingsPerNamespace.erase(currentNamespace());
|
|
|
|
|
m_enteredNamespaces.pop_back();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void NSCheckerVisitor::endVisit(TranslationUnitAST *)
|
|
|
|
|
{
|
|
|
|
|
// the last node, create the final result
|
|
|
|
|
// we must handle like the following: We search for foo::bar and have:
|
|
|
|
|
// using namespace foo::bar;
|
|
|
|
|
// namespace foo {
|
|
|
|
|
// // cursor/symbolPos here
|
|
|
|
|
// }
|
|
|
|
|
if (m_remainingNamespaces.empty()) {
|
|
|
|
|
// we are already finished
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
// find the longest combination of normal namespaces + using statements
|
|
|
|
|
int longestNamespaceList = 0;
|
|
|
|
|
int enteredNamespaceCount = 0;
|
|
|
|
|
// check 'using namespace ...;' statements in the global scope
|
|
|
|
|
const auto namespaces = m_usingsPerNamespace.find(nullptr);
|
|
|
|
|
if (namespaces != m_usingsPerNamespace.end())
|
|
|
|
|
longestNamespaceList = namespaces->second.length();
|
|
|
|
|
|
|
|
|
|
for (auto ns : m_enteredNamespaces) {
|
|
|
|
|
++enteredNamespaceCount;
|
|
|
|
|
const auto namespaces = m_usingsPerNamespace.find(ns);
|
|
|
|
|
int newListLength = enteredNamespaceCount;
|
|
|
|
|
if (namespaces != m_usingsPerNamespace.end())
|
|
|
|
|
newListLength += namespaces->second.length();
|
|
|
|
|
longestNamespaceList = std::max(newListLength, longestNamespaceList);
|
|
|
|
|
}
|
|
|
|
|
m_remainingNamespaces.erase(m_remainingNamespaces.begin(),
|
|
|
|
|
m_remainingNamespaces.begin() + longestNamespaceList
|
|
|
|
|
- m_enteredNamespaces.size());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString NSCheckerVisitor::getName(NamespaceAST *ns)
|
|
|
|
|
{
|
|
|
|
|
const Identifier *const id = translationUnit()->identifier(ns->identifier_token);
|
|
|
|
|
if (id)
|
|
|
|
|
return QString::fromUtf8(id->chars(), id->size());
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NamespaceAST *NSCheckerVisitor::currentNamespace()
|
|
|
|
|
{
|
|
|
|
|
return m_enteredNamespaces.empty() ? nullptr : m_enteredNamespaces.back();
|
|
|
|
|
}
|
|
|
|
|
|
2021-05-07 16:10:07 +02:00
|
|
|
ProjectExplorer::Project *projectForProjectPart(const ProjectPart &part)
|
|
|
|
|
{
|
2023-02-14 15:47:22 +01:00
|
|
|
return ProjectExplorer::ProjectManager::projectWithProjectFilePath(part.topLevelProject);
|
2021-05-07 16:10:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ProjectExplorer::Project *projectForProjectInfo(const ProjectInfo &info)
|
|
|
|
|
{
|
2023-02-14 15:47:22 +01:00
|
|
|
return ProjectExplorer::ProjectManager::projectWithProjectFilePath(info.projectFilePath());
|
2021-05-07 16:10:07 +02:00
|
|
|
}
|
|
|
|
|
|
2022-09-06 17:52:22 +02:00
|
|
|
void openEditor(const Utils::FilePath &filePath, bool inNextSplit, Utils::Id editorId)
|
2022-05-18 15:16:40 +02:00
|
|
|
{
|
|
|
|
|
using Core::EditorManager;
|
2022-09-06 17:52:22 +02:00
|
|
|
EditorManager::openEditor(filePath, editorId, inNextSplit ? EditorManager::OpenInOtherSplit
|
|
|
|
|
: EditorManager::NoFlags);
|
2022-05-18 15:16:40 +02:00
|
|
|
}
|
|
|
|
|
|
2023-05-31 14:29:03 +02:00
|
|
|
bool preferLowerCaseFileNames(ProjectExplorer::Project *project)
|
2022-09-21 13:38:42 +02:00
|
|
|
{
|
2023-05-31 14:29:03 +02:00
|
|
|
return Internal::CppEditorPlugin::fileSettings(project).lowerCaseFiles;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString preferredCxxHeaderSuffix(ProjectExplorer::Project *project)
|
|
|
|
|
{
|
|
|
|
|
return Internal::CppEditorPlugin::fileSettings(project).headerSuffix;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString preferredCxxSourceSuffix(ProjectExplorer::Project *project)
|
|
|
|
|
{
|
|
|
|
|
return Internal::CppEditorPlugin::fileSettings(project).sourceSuffix;
|
2022-09-21 13:38:42 +02:00
|
|
|
}
|
|
|
|
|
|
2023-08-23 17:26:02 +02:00
|
|
|
SearchResultItems symbolOccurrencesInDeclarationComments(
|
|
|
|
|
const Utils::SearchResultItems &symbolOccurrencesInCode)
|
|
|
|
|
{
|
|
|
|
|
if (symbolOccurrencesInCode.isEmpty())
|
|
|
|
|
return {};
|
|
|
|
|
|
|
|
|
|
// When using clangd, this function gets called every time the replacement string changes,
|
|
|
|
|
// so cache the results.
|
|
|
|
|
static QHash<SearchResultItems, SearchResultItems> resultCache;
|
|
|
|
|
if (const auto it = resultCache.constFind(symbolOccurrencesInCode);
|
|
|
|
|
it != resultCache.constEnd()) {
|
|
|
|
|
return it.value();
|
|
|
|
|
}
|
|
|
|
|
if (resultCache.size() > 5)
|
|
|
|
|
resultCache.clear();
|
|
|
|
|
|
|
|
|
|
QElapsedTimer timer;
|
|
|
|
|
timer.start();
|
|
|
|
|
Snapshot snapshot = CppModelManager::snapshot();
|
|
|
|
|
std::vector<std::unique_ptr<QTextDocument>> docPool;
|
|
|
|
|
using FileData = std::tuple<QTextDocument *, QString, Document::Ptr, QList<Token>>;
|
|
|
|
|
QHash<FilePath, FileData> dataPerFile;
|
|
|
|
|
QString symbolName;
|
|
|
|
|
const auto fileData = [&](const FilePath &filePath) -> FileData & {
|
|
|
|
|
auto &data = dataPerFile[filePath];
|
|
|
|
|
auto &[doc, content, cppDoc, allCommentTokens] = data;
|
|
|
|
|
if (!doc) {
|
|
|
|
|
if (TextEditor::TextDocument * const textDoc
|
|
|
|
|
= TextEditor::TextDocument::textDocumentForFilePath(filePath)) {
|
|
|
|
|
doc = textDoc->document();
|
|
|
|
|
} else {
|
|
|
|
|
std::unique_ptr<QTextDocument> newDoc = std::make_unique<QTextDocument>();
|
|
|
|
|
if (const auto content = TextFileFormat::readFile(
|
|
|
|
|
filePath, Core::EditorManager::defaultTextCodec())) {
|
|
|
|
|
newDoc->setPlainText(content.value());
|
|
|
|
|
}
|
|
|
|
|
doc = newDoc.get();
|
|
|
|
|
docPool.push_back(std::move(newDoc));
|
|
|
|
|
}
|
|
|
|
|
content = doc->toPlainText();
|
|
|
|
|
cppDoc = snapshot.preprocessedDocument(content.toUtf8(), filePath);
|
|
|
|
|
cppDoc->check();
|
|
|
|
|
}
|
|
|
|
|
return data;
|
|
|
|
|
};
|
|
|
|
|
static const auto addToken = [](QList<Token> &tokens, const Token &tok) {
|
|
|
|
|
if (!Utils::contains(tokens, [&tok](const Token &t) {
|
|
|
|
|
return t.byteOffset == tok.byteOffset; })) {
|
|
|
|
|
tokens << tok;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2023-09-22 15:20:32 +02:00
|
|
|
struct ClassInfo {
|
|
|
|
|
FilePath filePath;
|
|
|
|
|
int startOffset = -1;
|
|
|
|
|
int endOffset = -1;
|
|
|
|
|
};
|
|
|
|
|
std::optional<ClassInfo> classInfo;
|
|
|
|
|
|
2023-08-23 17:26:02 +02:00
|
|
|
// Collect comment blocks associated with replace locations.
|
|
|
|
|
for (const SearchResultItem &item : symbolOccurrencesInCode) {
|
|
|
|
|
const FilePath filePath = FilePath::fromUserInput(item.path().last());
|
|
|
|
|
auto &[doc, content, cppDoc, allCommentTokens] = fileData(filePath);
|
|
|
|
|
const Text::Range &range = item.mainRange();
|
|
|
|
|
if (symbolName.isEmpty()) {
|
|
|
|
|
const int symbolStartPos = Utils::Text::positionInText(doc, range.begin.line,
|
|
|
|
|
range.begin.column + 1);
|
|
|
|
|
const int symbolEndPos = Utils::Text::positionInText(doc, range.end.line,
|
|
|
|
|
range.end.column + 1);
|
|
|
|
|
symbolName = content.mid(symbolStartPos, symbolEndPos - symbolStartPos);
|
|
|
|
|
}
|
|
|
|
|
const QList<Token> commentTokens = commentsForDeclaration(symbolName, range.begin,
|
|
|
|
|
*doc, cppDoc);
|
|
|
|
|
for (const Token &tok : commentTokens)
|
|
|
|
|
addToken(allCommentTokens, tok);
|
|
|
|
|
|
2023-09-22 15:20:32 +02:00
|
|
|
if (!classInfo) {
|
2023-08-23 17:26:02 +02:00
|
|
|
QTextCursor cursor(doc);
|
|
|
|
|
cursor.setPosition(Text::positionInText(doc, range.begin.line, range.begin.column + 1));
|
2023-09-22 15:20:32 +02:00
|
|
|
Internal::CanonicalSymbol cs(cppDoc, snapshot);
|
|
|
|
|
Symbol * const canonicalSymbol = cs(cursor);
|
|
|
|
|
if (canonicalSymbol) {
|
|
|
|
|
classInfo.emplace();
|
|
|
|
|
if (Class * const klass = canonicalSymbol->asClass()) {
|
|
|
|
|
classInfo->filePath = canonicalSymbol->filePath();
|
|
|
|
|
classInfo->startOffset = klass->startOffset();
|
|
|
|
|
classInfo->endOffset = klass->endOffset();
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-08-23 17:26:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// We hook in between the end of the "regular" search and (possibly non-interactive)
|
|
|
|
|
// actions on it, so we must run synchronously in the UI thread and therefore be fast.
|
|
|
|
|
// If we notice we are lagging, just abort, as renaming the comments is not
|
|
|
|
|
// required for code correctness.
|
|
|
|
|
if (timer.elapsed() > 1000) {
|
|
|
|
|
resultCache.insert(symbolOccurrencesInCode, {});
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// If the symbol is a class, collect all comment blocks in the class body.
|
2023-09-22 15:20:32 +02:00
|
|
|
if (classInfo && !classInfo->filePath.isEmpty()) {
|
|
|
|
|
auto &[_1, _2, symbolCppDoc, commentTokens] = fileData(classInfo->filePath);
|
2023-08-23 17:26:02 +02:00
|
|
|
TranslationUnit * const tu = symbolCppDoc->translationUnit();
|
|
|
|
|
for (int i = 0; i < tu->commentCount(); ++i) {
|
|
|
|
|
const Token &tok = tu->commentAt(i);
|
2023-09-22 15:20:32 +02:00
|
|
|
if (tok.bytesBegin() < classInfo->startOffset)
|
2023-08-23 17:26:02 +02:00
|
|
|
continue;
|
2023-09-22 15:20:32 +02:00
|
|
|
if (tok.bytesBegin() >= classInfo->endOffset)
|
2023-08-23 17:26:02 +02:00
|
|
|
break;
|
|
|
|
|
addToken(commentTokens, tok);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Create new replace items for occurrences of the symbol name in collected comment blocks.
|
|
|
|
|
SearchResultItems commentItems;
|
|
|
|
|
for (auto it = dataPerFile.cbegin(); it != dataPerFile.cend(); ++it) {
|
|
|
|
|
const auto &[doc, content, cppDoc, commentTokens] = it.value();
|
|
|
|
|
const QStringView docView(content);
|
|
|
|
|
for (const Token &tok : commentTokens) {
|
|
|
|
|
const int tokenStartPos = cppDoc->translationUnit()->getTokenPositionInDocument(
|
|
|
|
|
tok, doc);
|
|
|
|
|
const int tokenEndPos = cppDoc->translationUnit()->getTokenEndPositionInDocument(
|
|
|
|
|
tok, doc);
|
|
|
|
|
const QStringView tokenView = docView.mid(tokenStartPos, tokenEndPos - tokenStartPos);
|
|
|
|
|
const QList<Text::Range> ranges = symbolOccurrencesInText(
|
|
|
|
|
*doc, tokenView, tokenStartPos, symbolName);
|
|
|
|
|
for (const Text::Range &range : ranges) {
|
|
|
|
|
SearchResultItem item;
|
|
|
|
|
item.setUseTextEditorFont(true);
|
|
|
|
|
item.setFilePath(it.key());
|
|
|
|
|
item.setMainRange(range);
|
|
|
|
|
item.setLineText(doc->findBlockByNumber(range.begin.line - 1).text());
|
|
|
|
|
commentItems << item;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
resultCache.insert(symbolOccurrencesInCode, commentItems);
|
|
|
|
|
return commentItems;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QList<Text::Range> symbolOccurrencesInText(const QTextDocument &doc, QStringView text, int offset,
|
|
|
|
|
const QString &symbolName)
|
|
|
|
|
{
|
|
|
|
|
QList<Text::Range> ranges;
|
|
|
|
|
int index = 0;
|
|
|
|
|
while (true) {
|
|
|
|
|
index = text.indexOf(symbolName, index);
|
|
|
|
|
if (index == -1)
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
// Prevent substring matching.
|
|
|
|
|
const auto checkAdjacent = [&](int i) {
|
|
|
|
|
if (i == -1 || i == text.size())
|
|
|
|
|
return true;
|
|
|
|
|
const QChar c = text.at(i);
|
|
|
|
|
if (c.isLetterOrNumber() || c == '_') {
|
|
|
|
|
index += symbolName.length();
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
};
|
|
|
|
|
if (!checkAdjacent(index - 1))
|
|
|
|
|
continue;
|
|
|
|
|
if (!checkAdjacent(index + symbolName.length()))
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
const Text::Position startPos = Text::Position::fromPositionInDocument(&doc, offset + index);
|
|
|
|
|
index += symbolName.length();
|
|
|
|
|
const Text::Position endPos = Text::Position::fromPositionInDocument(&doc, offset + index);
|
|
|
|
|
ranges << Text::Range{startPos, endPos};
|
|
|
|
|
}
|
|
|
|
|
return ranges;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QList<Text::Range> symbolOccurrencesInDeclarationComments(CppEditorWidget *editorWidget,
|
|
|
|
|
const QTextCursor &cursor)
|
|
|
|
|
{
|
|
|
|
|
if (!editorWidget)
|
|
|
|
|
return {};
|
|
|
|
|
const SemanticInfo &semanticInfo = editorWidget->semanticInfo();
|
|
|
|
|
const Document::Ptr &cppDoc = semanticInfo.doc;
|
|
|
|
|
if (!cppDoc)
|
|
|
|
|
return {};
|
2023-09-22 15:20:32 +02:00
|
|
|
Internal::CanonicalSymbol cs(cppDoc, semanticInfo.snapshot);
|
|
|
|
|
const Symbol * const symbol = cs(cursor);
|
2023-08-23 17:26:02 +02:00
|
|
|
if (!symbol || !symbol->asArgument())
|
|
|
|
|
return {};
|
|
|
|
|
const QTextDocument * const textDoc = editorWidget->textDocument()->document();
|
|
|
|
|
QTC_ASSERT(textDoc, return {});
|
|
|
|
|
const QList<Token> comments = commentsForDeclaration(symbol, *textDoc, cppDoc);
|
|
|
|
|
if (comments.isEmpty())
|
|
|
|
|
return {};
|
|
|
|
|
QList<Text::Range> ranges;
|
|
|
|
|
const QString &content = textDoc->toPlainText();
|
|
|
|
|
const QStringView docView = QStringView(content);
|
|
|
|
|
const QString symbolName = Overview().prettyName(symbol->name());
|
|
|
|
|
for (const Token &tok : comments) {
|
|
|
|
|
const int tokenStartPos = cppDoc->translationUnit()->getTokenPositionInDocument(
|
|
|
|
|
tok, textDoc);
|
|
|
|
|
const int tokenEndPos = cppDoc->translationUnit()->getTokenEndPositionInDocument(
|
|
|
|
|
tok, textDoc);
|
|
|
|
|
const QStringView tokenView = docView.mid(tokenStartPos, tokenEndPos - tokenStartPos);
|
|
|
|
|
ranges << symbolOccurrencesInText(*textDoc, tokenView, tokenStartPos, symbolName);
|
|
|
|
|
}
|
|
|
|
|
return ranges;
|
|
|
|
|
}
|
|
|
|
|
|
2021-09-02 08:23:46 +02:00
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
|
|
void decorateCppEditor(TextEditor::TextEditorWidget *editor)
|
|
|
|
|
{
|
|
|
|
|
editor->textDocument()->setSyntaxHighlighter(new CppHighlighter);
|
|
|
|
|
editor->textDocument()->setIndenter(
|
|
|
|
|
new CppQtStyleIndenter(editor->textDocument()->document()));
|
|
|
|
|
editor->setAutoCompleter(new CppAutoCompleter);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace Internal
|
2021-08-30 10:58:08 +02:00
|
|
|
} // CppEditor
|