CppEditor: Add quickfix for converting a function call

... to a Qt meta-method invocation.

Fixes: QTCREATORBUG-15972
Change-Id: Id84c83c5832cef32a877a451b0931ec47d2afe9d
Reviewed-by: Christian Stenger <christian.stenger@qt.io>
This commit is contained in:
Christian Kandeler
2024-02-05 16:57:56 +01:00
parent ef6f8df26d
commit e2756fde8b
4 changed files with 267 additions and 25 deletions

View File

@@ -9468,4 +9468,91 @@ void QuickfixTest::testMoveComments()
QuickFixOperationTest(documents, &factory, {}, {}, failMessage);
}
void QuickfixTest::testConvertToMetaMethodInvocation_data()
{
QTest::addColumn<QByteArray>("input");
QTest::addColumn<QByteArray>("expected");
// ^ marks the cursor locations.
// $ marks the replacement regions.
// The quoted string in the comment is the data tag.
// The rest of the comment is the replacement string.
const QByteArray allCases = R"(
class C {
public:
C() {
$this->^aSignal()$; // "signal from region on pointer to object" QMetaObject::invokeMethod(this, "aSignal")
C c;
$c.^aSignal()$; // "signal from region on object value" QMetaObject::invokeMethod(&c, "aSignal")
$(new C)->^aSignal()$; // "signal from region on expression" QMetaObject::invokeMethod((new C), "aSignal")
$emit this->^aSignal()$; // "signal from region, with emit" QMetaObject::invokeMethod(this, "aSignal")
$Q_EMIT this->^aSignal()$; // "signal from region, with Q_EMIT" QMetaObject::invokeMethod(this, "aSignal")
$this->^aSlot()$; // "slot from region" QMetaObject::invokeMethod(this, "aSlot")
$this->^noArgs()$; // "Q_SIGNAL, no arguments" QMetaObject::invokeMethod(this, "noArgs")
$this->^oneArg(0)$; // "Q_SLOT, one argument" QMetaObject::invokeMethod(this, "oneArg", Q_ARG(int, 0))
$this->^twoArgs(0, c)$; // "Q_INVOKABLE, two arguments" QMetaObject::invokeMethod(this, "twoArgs", Q_ARG(int, 0), Q_ARG(C, c))
this->^notInvokable(); // "not invokable"
}
signals:
void aSignal();
private slots:
void aSlot();
private:
Q_SIGNAL void noArgs();
Q_SLOT void oneArg(int index);
Q_INVOKABLE void twoArgs(int index, const C &value);
void notInvokable();
};
)";
qsizetype nextCursor = allCases.indexOf('^');
while (nextCursor != -1) {
const int commentStart = allCases.indexOf("//", nextCursor);
QVERIFY(commentStart != -1);
const int tagStart = allCases.indexOf('"', commentStart + 2);
QVERIFY(tagStart != -1);
const int tagEnd = allCases.indexOf('"', tagStart + 1);
QVERIFY(tagEnd != -1);
QByteArray input = allCases;
QByteArray output = allCases;
input.replace(nextCursor, 1, "@");
const QByteArray tag = allCases.mid(tagStart + 1, tagEnd - tagStart - 1);
const int prevNewline = allCases.lastIndexOf('\n', nextCursor);
const int regionStart = allCases.lastIndexOf('$', nextCursor);
bool hasReplacement = false;
if (regionStart != -1 && regionStart > prevNewline) {
const int regionEnd = allCases.indexOf('$', regionStart + 1);
QVERIFY(regionEnd != -1);
const int nextNewline = allCases.indexOf('\n', tagEnd);
QVERIFY(nextNewline != -1);
const QByteArray replacement
= allCases.mid(tagEnd + 1, nextNewline - tagEnd - 1).trimmed();
output.replace(regionStart, regionEnd - regionStart, replacement);
hasReplacement = true;
}
static const auto matcher = [](char c) { return c == '^' || c == '$'; };
input.removeIf(matcher);
if (hasReplacement) {
output.removeIf(matcher);
output.prepend("#include <QMetaObject>\n\n");
} else {
output.clear();
}
QTest::newRow(tag.data()) << input << output;
nextCursor = allCases.indexOf('^', nextCursor + 1);
}
}
void QuickfixTest::testConvertToMetaMethodInvocation()
{
QFETCH(QByteArray, input);
QFETCH(QByteArray, expected);
ConvertToMetaMethodCall factory;
QuickFixOperationTest({CppTestDocument::create("file.cpp", input, expected)}, &factory);
}
} // namespace CppEditor::Internal::Tests

View File

@@ -224,6 +224,9 @@ private slots:
void testMoveComments_data();
void testMoveComments();
void testConvertToMetaMethodInvocation_data();
void testConvertToMetaMethodInvocation();
};
} // namespace Tests

View File

@@ -221,8 +221,10 @@ Namespace *isNamespaceFunction(const LookupContext &context, Function *function)
}
// Given include is e.g. "afile.h" or <afile.h> (quotes/angle brackets included!).
void insertNewIncludeDirective(const QString &include, CppRefactoringFilePtr file,
const Document::Ptr &cppDocument)
static void insertNewIncludeDirective(const QString &include,
CppRefactoringFilePtr file,
const Document::Ptr &cppDocument,
ChangeSet &changes)
{
// Find optimal position
unsigned newLinesToPrepend = 0;
@@ -245,10 +247,7 @@ void insertNewIncludeDirective(const QString &include, CppRefactoringFilePtr fil
const QString textToInsert = prependedNewLines + includeLine + appendedNewLines;
// Insert
ChangeSet changes;
changes.insert(insertPosition, textToInsert);
file->setChangeSet(changes);
file->apply();
}
bool nameIncludesOperatorName(const Name *name)
@@ -322,22 +321,11 @@ QString nameString(const NameAST *name)
return CppCodeStyleSettings::currentProjectCodeStyleOverview().prettyName(name->name);
}
// FIXME: Needs to consider the scope at the insertion site.
QString declFromExpr(const TypeOrExpr &typeOrExpr, const CallAST *call, const NameAST *varName,
const Snapshot &snapshot, const LookupContext &context,
const CppRefactoringFilePtr &file, bool makeConst)
static FullySpecifiedType typeOfExpr(const ExpressionAST *expr,
const CppRefactoringFilePtr &file,
const Snapshot &snapshot,
const LookupContext &context)
{
const auto getTypeFromUser = [varName, call]() -> QString {
if (call)
return {};
const QString typeFromUser = QInputDialog::getText(Core::ICore::dialogParent(),
Tr::tr("Provide the type"),
Tr::tr("Data type:"), QLineEdit::Normal);
if (!typeFromUser.isEmpty())
return typeFromUser + ' ' + nameString(varName);
return {};
};
const auto getTypeOfExpr = [&](const ExpressionAST *expr) -> FullySpecifiedType {
TypeOfExpression typeOfExpression;
typeOfExpression.init(file->cppDocument(), snapshot, context.bindings());
Scope *scope = file->scopeAt(expr->firstToken());
@@ -357,6 +345,25 @@ QString declFromExpr(const TypeOrExpr &typeOrExpr, const CallAST *call, const Na
Control *control = context.bindings()->control().get();
return rewriteType(result.first().type(), &env, control);
}
// FIXME: Needs to consider the scope at the insertion site.
QString declFromExpr(const TypeOrExpr &typeOrExpr, const CallAST *call, const NameAST *varName,
const Snapshot &snapshot, const LookupContext &context,
const CppRefactoringFilePtr &file, bool makeConst)
{
const auto getTypeFromUser = [varName, call]() -> QString {
if (call)
return {};
const QString typeFromUser = QInputDialog::getText(Core::ICore::dialogParent(),
Tr::tr("Provide the type"),
Tr::tr("Data type:"), QLineEdit::Normal);
if (!typeFromUser.isEmpty())
return typeFromUser + ' ' + nameString(varName);
return {};
};
const auto getTypeOfExpr = [&](const ExpressionAST *expr) -> FullySpecifiedType {
return typeOfExpr(expr, file, snapshot, context);
};
const Overview oo = CppCodeStyleSettings::currentProjectCodeStyleOverview();
@@ -1791,7 +1798,10 @@ void AddIncludeForUndefinedIdentifierOp::perform()
CppRefactoringChanges refactoring(snapshot());
CppRefactoringFilePtr file = refactoring.cppFile(filePath());
insertNewIncludeDirective(m_include, file, semanticInfo().doc);
ChangeSet changes;
insertNewIncludeDirective(m_include, file, semanticInfo().doc, changes);
file->setChangeSet(changes);
file->apply();
}
AddForwardDeclForUndefinedIdentifierOp::AddForwardDeclForUndefinedIdentifierOp(
@@ -9801,6 +9811,138 @@ void MoveFunctionComments::doMatch(const CppQuickFixInterface &interface,
}
}
namespace {
class ConvertToMetaMethodCallOp : public CppQuickFixOperation
{
public:
ConvertToMetaMethodCallOp(const CppQuickFixInterface &interface, CallAST *callAst)
: CppQuickFixOperation(interface), m_callAst(callAst)
{
setDescription(Tr::tr("Convert function call to Qt meta-method invocation"));
}
private:
void perform() override
{
// Construct the argument list.
Overview ov;
QStringList arguments;
for (ExpressionListAST *it = m_callAst->expression_list; it; it = it->next) {
if (!it->value)
return;
const FullySpecifiedType argType
= typeOfExpr(it->value, currentFile(), snapshot(), context());
if (!argType.isValid())
return;
arguments << QString::fromUtf8("Q_ARG(%1, %2)")
.arg(ov.prettyType(argType), currentFile()->textOf(it->value));
}
QString argsString = arguments.join(", ");
if (!argsString.isEmpty())
argsString.prepend(", ");
// Construct the replace string.
const auto memberAccessAst = m_callAst->base_expression->asMemberAccess();
QTC_ASSERT(memberAccessAst, return);
QString baseExpr = currentFile()->textOf(memberAccessAst->base_expression);
const FullySpecifiedType baseExprType
= typeOfExpr(memberAccessAst->base_expression, currentFile(), snapshot(), context());
if (!baseExprType.isValid())
return;
if (!baseExprType->asPointerType())
baseExpr.prepend('&');
const QString functionName = currentFile()->textOf(memberAccessAst->member_name);
const QString qMetaObject = "QMetaObject";
const QString newCall = QString::fromUtf8("%1::invokeMethod(%2, \"%3\"%4)")
.arg(qMetaObject, baseExpr, functionName, argsString);
// Determine the start and end positions of the replace operation.
// If the call is preceded by an "emit" keyword, that one has to be removed as well.
int firstToken = m_callAst->firstToken();
if (firstToken > 0)
switch (semanticInfo().doc->translationUnit()->tokenKind(firstToken - 1)) {
case T_EMIT: case T_Q_EMIT: --firstToken; break;
default: break;
}
const TranslationUnit *const tu = semanticInfo().doc->translationUnit();
const int startPos = tu->getTokenPositionInDocument(firstToken, textDocument());
const int endPos = tu->getTokenPositionInDocument(m_callAst->lastToken(), textDocument());
// Replace the old call with the new one.
ChangeSet changes;
changes.replace(startPos, endPos, newCall);
// Insert include for QMetaObject, if necessary.
const Identifier qMetaObjectId(qPrintable(qMetaObject), qMetaObject.size());
Scope * const scope = currentFile()->scopeAt(firstToken);
const QList<LookupItem> results = context().lookup(&qMetaObjectId, scope);
bool isDeclared = false;
for (const LookupItem &item : results) {
if (Symbol *declaration = item.declaration(); declaration && declaration->asClass()) {
isDeclared = true;
break;
}
}
if (!isDeclared) {
insertNewIncludeDirective('<' + qMetaObject + '>', currentFile(), semanticInfo().doc,
changes);
}
// Apply the changes.
currentFile()->setChangeSet(changes);
currentFile()->apply();
}
const CallAST * const m_callAst;
};
} // namespace
void ConvertToMetaMethodCall::doMatch(const CppQuickFixInterface &interface,
TextEditor::QuickFixOperations &result)
{
const Document::Ptr &cppDoc = interface.currentFile()->cppDocument();
const QList<AST *> path = ASTPath(cppDoc)(interface.cursor());
if (path.isEmpty())
return;
// Are we on a member function call?
CallAST *callAst = nullptr;
for (auto it = path.crbegin(); it != path.crend(); ++it) {
if ((callAst = (*it)->asCall()))
break;
}
if (!callAst || !callAst->base_expression)
return;
ExpressionAST *baseExpr = nullptr;
const NameAST *nameAst = nullptr;
if (const MemberAccessAST * const ast = callAst->base_expression->asMemberAccess()) {
baseExpr = ast->base_expression;
nameAst = ast->member_name;
}
if (!baseExpr || !nameAst || !nameAst->name)
return;
// Locate called function and check whether it is invokable.
Scope *scope = cppDoc->globalNamespace();
for (auto it = path.crbegin(); it != path.crend(); ++it) {
if (const CompoundStatementAST * const stmtAst = (*it)->asCompoundStatement()) {
scope = stmtAst->symbol;
break;
}
}
const LookupContext context(cppDoc, interface.snapshot());
TypeOfExpression exprType;
exprType.setExpandTemplates(true);
exprType.init(cppDoc, interface.snapshot());
const QList<LookupItem> typeMatches = exprType(callAst->base_expression, cppDoc, scope);
for (const LookupItem &item : typeMatches) {
if (const auto func = item.type()->asFunctionType(); func && func->methodKey()) {
result << new ConvertToMetaMethodCallOp(interface, callAst);
return;
}
}
}
void createCppQuickFixes()
{
new AddIncludeForUndefinedIdentifier;
@@ -9860,6 +10002,7 @@ void createCppQuickFixes()
new GenerateConstructor;
new ConvertCommentStyle;
new MoveFunctionComments;
new ConvertToMetaMethodCall;
}
void destroyCppQuickFixes()

View File

@@ -617,5 +617,14 @@ private:
TextEditor::QuickFixOperations &result) override;
};
//! Converts a normal function call into a meta method invocation, if the functions is
//! marked as invokable.
class ConvertToMetaMethodCall : public CppQuickFixFactory
{
private:
void doMatch(const CppQuickFixInterface &interface,
TextEditor::QuickFixOperations &result) override;
};
} // namespace Internal
} // namespace CppEditor