forked from qt-creator/qt-creator
Clang: Refactor ClangCompletionContextAnalyzer
Change-Id: Ib42ddc672da8b068591129e2e0b9652d3e07ad58 Reviewed-by: Nikolai Kosjar <nikolai.kosjar@theqtcompany.com>
This commit is contained in:
committed by
Nikolai Kosjar
parent
5a791e8839
commit
ae5d92d618
@@ -0,0 +1,249 @@
|
|||||||
|
/****************************************************************************
|
||||||
|
**
|
||||||
|
** Copyright (C) 2015 The Qt Company Ltd.
|
||||||
|
** Contact: http://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 http://www.qt.io/terms-conditions. For further information
|
||||||
|
** use the contact form at http://www.qt.io/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 or version 3 as published by the Free
|
||||||
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
||||||
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
||||||
|
** following information to ensure the GNU Lesser General Public License
|
||||||
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
||||||
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
||||||
|
**
|
||||||
|
** In addition, as a special exception, The Qt Company gives you certain additional
|
||||||
|
** rights. These rights are described in The Qt Company LGPL Exception
|
||||||
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
||||||
|
**
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
#include "activationsequencecontextprocessor.h"
|
||||||
|
|
||||||
|
#include "activationsequenceprocessor.h"
|
||||||
|
|
||||||
|
#include <cplusplus/BackwardsScanner.h>
|
||||||
|
#include <cplusplus/ExpressionUnderCursor.h>
|
||||||
|
#include <cplusplus/SimpleLexer.h>
|
||||||
|
|
||||||
|
#include <QRegExp>
|
||||||
|
#include <QTextDocument>
|
||||||
|
|
||||||
|
namespace ClangCodeModel {
|
||||||
|
namespace Internal {
|
||||||
|
|
||||||
|
ActivationSequenceContextProcessor::ActivationSequenceContextProcessor(const ClangCompletionAssistInterface *assistInterface)
|
||||||
|
: m_textCursor(assistInterface->textDocument()),
|
||||||
|
m_assistInterface(assistInterface),
|
||||||
|
m_positionInDocument(assistInterface->position()),
|
||||||
|
m_positionAfterOperator(m_positionInDocument),
|
||||||
|
m_positionBeforeOperator(m_positionAfterOperator)
|
||||||
|
|
||||||
|
{
|
||||||
|
m_textCursor.setPosition(m_positionInDocument);
|
||||||
|
|
||||||
|
process();
|
||||||
|
}
|
||||||
|
|
||||||
|
CPlusPlus::Kind ActivationSequenceContextProcessor::completionKind() const
|
||||||
|
{
|
||||||
|
return m_completionKind;
|
||||||
|
}
|
||||||
|
|
||||||
|
const QTextCursor &ActivationSequenceContextProcessor::textCursor_forTestOnly() const
|
||||||
|
{
|
||||||
|
return m_textCursor;
|
||||||
|
}
|
||||||
|
|
||||||
|
int ActivationSequenceContextProcessor::positionAfterOperator() const
|
||||||
|
{
|
||||||
|
return m_positionAfterOperator;
|
||||||
|
}
|
||||||
|
|
||||||
|
int ActivationSequenceContextProcessor::positionBeforeOperator() const
|
||||||
|
{
|
||||||
|
return m_positionBeforeOperator;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ActivationSequenceContextProcessor::process()
|
||||||
|
{
|
||||||
|
skipeWhiteSpacesAndIdentifierBeforeCursor();
|
||||||
|
processActivationSequence();
|
||||||
|
|
||||||
|
if (m_completionKind != CPlusPlus::T_EOF_SYMBOL) {
|
||||||
|
processStringLiteral();
|
||||||
|
processComma();
|
||||||
|
generateTokens();
|
||||||
|
processDoxygenComment();
|
||||||
|
processComment();
|
||||||
|
processInclude();
|
||||||
|
processSlashOutsideOfAString();
|
||||||
|
processLeftParen();
|
||||||
|
processPreprocessorInclude();
|
||||||
|
resetPositionForEOFCompletionKind();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ActivationSequenceContextProcessor::processActivationSequence()
|
||||||
|
{
|
||||||
|
const auto activationSequence = m_assistInterface->textAt(m_positionInDocument - 3, 3);
|
||||||
|
ActivationSequenceProcessor activationSequenceProcessor(activationSequence,
|
||||||
|
m_positionInDocument,
|
||||||
|
true);
|
||||||
|
|
||||||
|
m_completionKind = activationSequenceProcessor.completionKind();
|
||||||
|
m_positionBeforeOperator = activationSequenceProcessor.position();
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void ActivationSequenceContextProcessor::processStringLiteral()
|
||||||
|
{
|
||||||
|
if (m_completionKind == CPlusPlus::T_STRING_LITERAL) {
|
||||||
|
QTextCursor selectionTextCursor = m_textCursor;
|
||||||
|
selectionTextCursor.movePosition(QTextCursor::StartOfLine, QTextCursor::KeepAnchor);
|
||||||
|
QString selection = selectionTextCursor.selectedText();
|
||||||
|
if (selection.indexOf(QLatin1Char('"')) < selection.length() - 1)
|
||||||
|
m_completionKind = CPlusPlus::T_EOF_SYMBOL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ActivationSequenceContextProcessor::processComma()
|
||||||
|
{
|
||||||
|
if (m_completionKind == CPlusPlus::T_COMMA) {
|
||||||
|
CPlusPlus::ExpressionUnderCursor expressionUnderCursor(m_assistInterface->languageFeatures());
|
||||||
|
if (expressionUnderCursor.startOfFunctionCall(m_textCursor) == -1)
|
||||||
|
m_completionKind = CPlusPlus::T_EOF_SYMBOL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ActivationSequenceContextProcessor::generateTokens()
|
||||||
|
{
|
||||||
|
CPlusPlus::SimpleLexer tokenize;
|
||||||
|
tokenize.setLanguageFeatures(m_assistInterface->languageFeatures());
|
||||||
|
tokenize.setSkipComments(false);
|
||||||
|
auto state = CPlusPlus::BackwardsScanner::previousBlockState(m_textCursor.block());
|
||||||
|
m_tokens = tokenize(m_textCursor.block().text(), state);
|
||||||
|
int leftOfCursorTokenIndex = std::max(0, m_textCursor.positionInBlock() - 1);
|
||||||
|
m_tokenIndex= CPlusPlus::SimpleLexer::tokenBefore(m_tokens, leftOfCursorTokenIndex); // get the token at the left of the cursor
|
||||||
|
if (m_tokenIndex > -1)
|
||||||
|
m_token = m_tokens.at(m_tokenIndex);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ActivationSequenceContextProcessor::processDoxygenComment()
|
||||||
|
{
|
||||||
|
if (m_completionKind == CPlusPlus::T_DOXY_COMMENT
|
||||||
|
&& !(m_token.is(CPlusPlus::T_DOXY_COMMENT)
|
||||||
|
|| m_token.is(CPlusPlus::T_CPP_DOXY_COMMENT)))
|
||||||
|
m_completionKind = CPlusPlus::T_EOF_SYMBOL;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ActivationSequenceContextProcessor::processComment()
|
||||||
|
{
|
||||||
|
if (m_token.is(CPlusPlus::T_COMMENT) || m_token.is(CPlusPlus::T_CPP_COMMENT))
|
||||||
|
m_completionKind = CPlusPlus::T_EOF_SYMBOL;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ActivationSequenceContextProcessor::processInclude()
|
||||||
|
{
|
||||||
|
if (m_token.isLiteral() && !isCompletionKindStringLiteralOrSlash())
|
||||||
|
m_completionKind = CPlusPlus::T_EOF_SYMBOL;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ActivationSequenceContextProcessor::processSlashOutsideOfAString()
|
||||||
|
{
|
||||||
|
if (m_completionKind ==CPlusPlus::T_SLASH
|
||||||
|
&& (m_token.isNot(CPlusPlus::T_STRING_LITERAL)
|
||||||
|
&& m_token.isNot(CPlusPlus::T_ANGLE_STRING_LITERAL)))
|
||||||
|
m_completionKind = CPlusPlus::T_EOF_SYMBOL;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ActivationSequenceContextProcessor::processLeftParen()
|
||||||
|
{
|
||||||
|
if (m_completionKind == CPlusPlus::T_LPAREN) {
|
||||||
|
if (m_tokenIndex > 0) {
|
||||||
|
// look at the token at the left of T_LPAREN
|
||||||
|
const CPlusPlus::Token &previousToken = m_tokens.at(m_tokenIndex - 1);
|
||||||
|
switch (previousToken.kind()) {
|
||||||
|
case CPlusPlus::T_IDENTIFIER:
|
||||||
|
case CPlusPlus::T_GREATER:
|
||||||
|
case CPlusPlus::T_SIGNAL:
|
||||||
|
case CPlusPlus::T_SLOT:
|
||||||
|
break; // good
|
||||||
|
|
||||||
|
default:
|
||||||
|
// that's a bad token :)
|
||||||
|
m_completionKind = CPlusPlus::T_EOF_SYMBOL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ActivationSequenceContextProcessor::isCompletionKindStringLiteralOrSlash() const
|
||||||
|
{
|
||||||
|
return m_completionKind == CPlusPlus::T_STRING_LITERAL
|
||||||
|
|| m_completionKind == CPlusPlus::T_ANGLE_STRING_LITERAL
|
||||||
|
|| m_completionKind == CPlusPlus::T_SLASH;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ActivationSequenceContextProcessor::isProbablyPreprocessorIncludeDirective() const
|
||||||
|
{
|
||||||
|
return m_tokens.size() >= 3
|
||||||
|
&& m_tokens.at(0).is(CPlusPlus::T_POUND)
|
||||||
|
&& m_tokens.at(1).is(CPlusPlus::T_IDENTIFIER)
|
||||||
|
&& (m_tokens.at(2).is(CPlusPlus::T_STRING_LITERAL)
|
||||||
|
|| m_tokens.at(2).is(CPlusPlus::T_ANGLE_STRING_LITERAL));
|
||||||
|
}
|
||||||
|
|
||||||
|
void ActivationSequenceContextProcessor::processPreprocessorInclude()
|
||||||
|
{
|
||||||
|
if (isCompletionKindStringLiteralOrSlash()) {
|
||||||
|
if (isProbablyPreprocessorIncludeDirective()) {
|
||||||
|
const CPlusPlus::Token &directiveToken = m_tokens.at(1);
|
||||||
|
QString directive = m_textCursor.block().text().mid(directiveToken.bytesBegin(),
|
||||||
|
directiveToken.bytes());
|
||||||
|
if (directive != QStringLiteral("include")
|
||||||
|
&& directive != QStringLiteral("include_next")
|
||||||
|
&& directive != QStringLiteral("import"))
|
||||||
|
m_completionKind = CPlusPlus::T_EOF_SYMBOL;
|
||||||
|
} else {
|
||||||
|
m_completionKind = CPlusPlus::T_EOF_SYMBOL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ActivationSequenceContextProcessor::resetPositionForEOFCompletionKind()
|
||||||
|
{
|
||||||
|
if (m_completionKind == CPlusPlus::T_EOF_SYMBOL)
|
||||||
|
m_positionBeforeOperator = m_positionInDocument;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ActivationSequenceContextProcessor::skipeWhiteSpacesAndIdentifierBeforeCursor()
|
||||||
|
{
|
||||||
|
QTextDocument *document = m_assistInterface->textDocument();
|
||||||
|
|
||||||
|
const QRegExp findNonWhiteSpaceRegularExpression(QStringLiteral("[^\\s\\w]"));
|
||||||
|
|
||||||
|
auto nonWhiteSpaceTextCursor = document->find(findNonWhiteSpaceRegularExpression,
|
||||||
|
m_positionInDocument,
|
||||||
|
QTextDocument::FindBackward);
|
||||||
|
|
||||||
|
if (!nonWhiteSpaceTextCursor.isNull()) {
|
||||||
|
m_positionInDocument = nonWhiteSpaceTextCursor.position();
|
||||||
|
m_positionAfterOperator = m_positionInDocument;
|
||||||
|
m_textCursor.setPosition(m_positionInDocument);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Internal
|
||||||
|
} // namespace ClangCodeModel
|
||||||
|
|
||||||
@@ -0,0 +1,92 @@
|
|||||||
|
/****************************************************************************
|
||||||
|
**
|
||||||
|
** Copyright (C) 2015 The Qt Company Ltd.
|
||||||
|
** Contact: http://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 http://www.qt.io/terms-conditions. For further information
|
||||||
|
** use the contact form at http://www.qt.io/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 or version 3 as published by the Free
|
||||||
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
||||||
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
||||||
|
** following information to ensure the GNU Lesser General Public License
|
||||||
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
||||||
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
||||||
|
**
|
||||||
|
** In addition, as a special exception, The Qt Company gives you certain additional
|
||||||
|
** rights. These rights are described in The Qt Company LGPL Exception
|
||||||
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
||||||
|
**
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
#ifndef CLANGCODEMODEL_INTERNAL_ACTIVATIONSEQUENCECONTEXTPROCESSOR_H
|
||||||
|
#define CLANGCODEMODEL_INTERNAL_ACTIVATIONSEQUENCECONTEXTPROCESSOR_H
|
||||||
|
|
||||||
|
#include <clangcodemodel/clangcompletionassistinterface.h>
|
||||||
|
|
||||||
|
#include <cplusplus/Token.h>
|
||||||
|
|
||||||
|
#include <QTextCursor>
|
||||||
|
|
||||||
|
QT_BEGIN_NAMESPACE
|
||||||
|
class QTextDocument;
|
||||||
|
QT_END_NAMESPACE
|
||||||
|
|
||||||
|
namespace ClangCodeModel {
|
||||||
|
namespace Internal {
|
||||||
|
|
||||||
|
class ActivationSequenceContextProcessor
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
ActivationSequenceContextProcessor(const ClangCompletionAssistInterface *assistInterface);
|
||||||
|
|
||||||
|
CPlusPlus::Kind completionKind() const;
|
||||||
|
|
||||||
|
const QTextCursor &textCursor_forTestOnly() const;
|
||||||
|
|
||||||
|
int positionAfterOperator() const;
|
||||||
|
int positionBeforeOperator() const;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void process();
|
||||||
|
void skipeWhiteSpacesAndIdentifierBeforeCursor();
|
||||||
|
void processActivationSequence();
|
||||||
|
void processStringLiteral();
|
||||||
|
void processComma();
|
||||||
|
void generateTokens();
|
||||||
|
void processDoxygenComment();
|
||||||
|
void processComment();
|
||||||
|
void processInclude();
|
||||||
|
void processSlashOutsideOfAString();
|
||||||
|
void processLeftParen();
|
||||||
|
void processPreprocessorInclude();
|
||||||
|
void resetPositionForEOFCompletionKind();
|
||||||
|
|
||||||
|
bool isCompletionKindStringLiteralOrSlash() const;
|
||||||
|
bool isProbablyPreprocessorIncludeDirective() const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
QVector<CPlusPlus::Token> m_tokens;
|
||||||
|
QTextCursor m_textCursor;
|
||||||
|
CPlusPlus::Token m_token;
|
||||||
|
const ClangCompletionAssistInterface *m_assistInterface;
|
||||||
|
int m_tokenIndex;
|
||||||
|
int m_positionInDocument;
|
||||||
|
int m_positionAfterOperator;
|
||||||
|
int m_positionBeforeOperator;
|
||||||
|
CPlusPlus::Kind m_completionKind;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Internal
|
||||||
|
} // namespace ClangCodeModel
|
||||||
|
|
||||||
|
#endif // CLANGCODEMODEL_INTERNAL_ACTIVATIONSEQUENCECONTEXTPROCESSOR_H
|
||||||
@@ -164,8 +164,8 @@ void ActivationSequenceProcessor::processArrowStar()
|
|||||||
|
|
||||||
void ActivationSequenceProcessor::processDoxyGenComment()
|
void ActivationSequenceProcessor::processDoxyGenComment()
|
||||||
{
|
{
|
||||||
if ((m_char2 == QLatin1Char('\\') || m_char2 == QLatin1Char('@'))
|
if ((m_char2.isNull() || m_char2.isSpace())
|
||||||
&& (m_char3.isNull() || m_char3.isSpace())) {
|
&& (m_char3 == QLatin1Char('\\') || m_char3 == QLatin1Char('@'))) {
|
||||||
m_completionKind = CPlusPlus::T_DOXY_COMMENT;
|
m_completionKind = CPlusPlus::T_DOXY_COMMENT;
|
||||||
m_offset = 1;
|
m_offset = 1;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ DEFINES += "\"CLANG_RESOURCE_DIR=\\\"$${LLVM_LIBDIR}/clang/$${LLVM_VERSION}/incl
|
|||||||
unix:QMAKE_LFLAGS += -Wl,-rpath,\'$$LLVM_LIBDIR\'
|
unix:QMAKE_LFLAGS += -Wl,-rpath,\'$$LLVM_LIBDIR\'
|
||||||
|
|
||||||
SOURCES += \
|
SOURCES += \
|
||||||
|
activationsequencecontextprocessor.cpp \
|
||||||
activationsequenceprocessor.cpp \
|
activationsequenceprocessor.cpp \
|
||||||
clangassistproposal.cpp \
|
clangassistproposal.cpp \
|
||||||
clangassistproposalitem.cpp \
|
clangassistproposalitem.cpp \
|
||||||
@@ -48,6 +49,7 @@ SOURCES += \
|
|||||||
|
|
||||||
|
|
||||||
HEADERS += \
|
HEADERS += \
|
||||||
|
activationsequencecontextprocessor.h \
|
||||||
activationsequenceprocessor.h \
|
activationsequenceprocessor.h \
|
||||||
clangassistproposal.h \
|
clangassistproposal.h \
|
||||||
clangassistproposalitem.h \
|
clangassistproposalitem.h \
|
||||||
@@ -107,13 +109,10 @@ equals(TEST, 1) {
|
|||||||
test/clang_tests_database.qrc
|
test/clang_tests_database.qrc
|
||||||
|
|
||||||
HEADERS += \
|
HEADERS += \
|
||||||
test/clangcodecompletion_test.h \
|
test/clangcodecompletion_test.h
|
||||||
test/clangcompletioncontextanalyzertest.h
|
|
||||||
|
|
||||||
SOURCES += \
|
SOURCES += \
|
||||||
test/clangcodecompletion_test.cpp \
|
test/clangcodecompletion_test.cpp
|
||||||
test/clangcompletioncontextanalyzertest.cpp
|
|
||||||
|
|
||||||
|
|
||||||
DISTFILES += \
|
DISTFILES += \
|
||||||
test/mysource.cpp \
|
test/mysource.cpp \
|
||||||
|
|||||||
@@ -53,6 +53,8 @@ QtcPlugin {
|
|||||||
name: "Completion support"
|
name: "Completion support"
|
||||||
condition: product.clangCompletion
|
condition: product.clangCompletion
|
||||||
files: [
|
files: [
|
||||||
|
"activationsequencecontextprocessor.cpp",
|
||||||
|
"activationsequencecontextprocessor.h",
|
||||||
"activationsequenceprocessor.cpp",
|
"activationsequenceprocessor.cpp",
|
||||||
"activationsequenceprocessor.h",
|
"activationsequenceprocessor.h",
|
||||||
"clangassistproposal.cpp",
|
"clangassistproposal.cpp",
|
||||||
@@ -104,8 +106,6 @@ QtcPlugin {
|
|||||||
"clang_tests_database.qrc",
|
"clang_tests_database.qrc",
|
||||||
"clangcodecompletion_test.cpp",
|
"clangcodecompletion_test.cpp",
|
||||||
"clangcodecompletion_test.h",
|
"clangcodecompletion_test.h",
|
||||||
"clangcompletioncontextanalyzertest.cpp",
|
|
||||||
"clangcompletioncontextanalyzertest.h",
|
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -36,7 +36,6 @@
|
|||||||
|
|
||||||
#ifdef WITH_TESTS
|
#ifdef WITH_TESTS
|
||||||
# include "test/clangcodecompletion_test.h"
|
# include "test/clangcodecompletion_test.h"
|
||||||
# include "test/clangcompletioncontextanalyzertest.h"
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include <cpptools/cppmodelmanager.h>
|
#include <cpptools/cppmodelmanager.h>
|
||||||
@@ -93,7 +92,6 @@ QList<QObject *> ClangCodeModelPlugin::createTestObjects() const
|
|||||||
{
|
{
|
||||||
return {
|
return {
|
||||||
new Tests::ClangCodeCompletionTest,
|
new Tests::ClangCodeCompletionTest,
|
||||||
new Tests::ClangCompletionContextAnalyzerTest
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -1,7 +1,11 @@
|
|||||||
INCLUDEPATH += $$PWD
|
INCLUDEPATH += $$PWD
|
||||||
|
|
||||||
SOURCES += $$PWD/completionchunkstotextconverter.cpp \
|
SOURCES += $$PWD/completionchunkstotextconverter.cpp \
|
||||||
$$PWD/activationsequenceprocessor.cpp
|
$$PWD/activationsequenceprocessor.cpp \
|
||||||
|
$$PWD/activationsequencecontextprocessor.cpp \
|
||||||
|
$$PWD/clangcompletioncontextanalyzer.cpp
|
||||||
|
|
||||||
HEADERS += $$PWD/completionchunkstotextconverter.h \
|
HEADERS += $$PWD/completionchunkstotextconverter.h \
|
||||||
$$PWD/activationsequenceprocessor.h
|
$$PWD/activationsequenceprocessor.h \
|
||||||
|
$$PWD/activationsequencecontextprocessor.h \
|
||||||
|
$$PWD/clangcompletioncontextanalyzer.h
|
||||||
|
|||||||
@@ -31,6 +31,9 @@
|
|||||||
|
|
||||||
#include "clangcompletioncontextanalyzer.h"
|
#include "clangcompletioncontextanalyzer.h"
|
||||||
|
|
||||||
|
#include "activationsequenceprocessor.h"
|
||||||
|
#include "activationsequencecontextprocessor.h"
|
||||||
|
|
||||||
#include <texteditor/codeassist/assistinterface.h>
|
#include <texteditor/codeassist/assistinterface.h>
|
||||||
|
|
||||||
#include <cplusplus/BackwardsScanner.h>
|
#include <cplusplus/BackwardsScanner.h>
|
||||||
@@ -47,80 +50,6 @@ using namespace CPlusPlus;
|
|||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
int activationSequenceChar(const QChar &ch, const QChar &ch2, const QChar &ch3,
|
|
||||||
unsigned *kind, bool wantFunctionCall)
|
|
||||||
{
|
|
||||||
int referencePosition = 0;
|
|
||||||
int completionKind = T_EOF_SYMBOL;
|
|
||||||
switch (ch.toLatin1()) {
|
|
||||||
case '.':
|
|
||||||
if (ch2 != QLatin1Char('.')) {
|
|
||||||
completionKind = T_DOT;
|
|
||||||
referencePosition = 1;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case ',':
|
|
||||||
completionKind = T_COMMA;
|
|
||||||
referencePosition = 1;
|
|
||||||
break;
|
|
||||||
case '(':
|
|
||||||
if (wantFunctionCall) {
|
|
||||||
completionKind = T_LPAREN;
|
|
||||||
referencePosition = 1;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case ':':
|
|
||||||
if (ch3 != QLatin1Char(':') && ch2 == QLatin1Char(':')) {
|
|
||||||
completionKind = T_COLON_COLON;
|
|
||||||
referencePosition = 2;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case '>':
|
|
||||||
if (ch2 == QLatin1Char('-')) {
|
|
||||||
completionKind = T_ARROW;
|
|
||||||
referencePosition = 2;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case '*':
|
|
||||||
if (ch2 == QLatin1Char('.')) {
|
|
||||||
completionKind = T_DOT_STAR;
|
|
||||||
referencePosition = 2;
|
|
||||||
} else if (ch3 == QLatin1Char('-') && ch2 == QLatin1Char('>')) {
|
|
||||||
completionKind = T_ARROW_STAR;
|
|
||||||
referencePosition = 3;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case '\\':
|
|
||||||
case '@':
|
|
||||||
if (ch2.isNull() || ch2.isSpace()) {
|
|
||||||
completionKind = T_DOXY_COMMENT;
|
|
||||||
referencePosition = 1;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case '<':
|
|
||||||
completionKind = T_ANGLE_STRING_LITERAL;
|
|
||||||
referencePosition = 1;
|
|
||||||
break;
|
|
||||||
case '"':
|
|
||||||
completionKind = T_STRING_LITERAL;
|
|
||||||
referencePosition = 1;
|
|
||||||
break;
|
|
||||||
case '/':
|
|
||||||
completionKind = T_SLASH;
|
|
||||||
referencePosition = 1;
|
|
||||||
break;
|
|
||||||
case '#':
|
|
||||||
completionKind = T_POUND;
|
|
||||||
referencePosition = 1;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (kind)
|
|
||||||
*kind = completionKind;
|
|
||||||
|
|
||||||
return referencePosition;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool isTokenForIncludePathCompletion(unsigned tokenKind)
|
bool isTokenForIncludePathCompletion(unsigned tokenKind)
|
||||||
{
|
{
|
||||||
return tokenKind == T_STRING_LITERAL
|
return tokenKind == T_STRING_LITERAL
|
||||||
@@ -143,7 +72,7 @@ namespace ClangCodeModel {
|
|||||||
namespace Internal {
|
namespace Internal {
|
||||||
|
|
||||||
ClangCompletionContextAnalyzer::ClangCompletionContextAnalyzer(
|
ClangCompletionContextAnalyzer::ClangCompletionContextAnalyzer(
|
||||||
const TextEditor::AssistInterface *assistInterface,
|
const ClangCompletionAssistInterface *assistInterface,
|
||||||
CPlusPlus::LanguageFeatures languageFeatures)
|
CPlusPlus::LanguageFeatures languageFeatures)
|
||||||
: m_interface(assistInterface)
|
: m_interface(assistInterface)
|
||||||
, m_languageFeatures(languageFeatures)
|
, m_languageFeatures(languageFeatures)
|
||||||
@@ -153,67 +82,20 @@ ClangCompletionContextAnalyzer::ClangCompletionContextAnalyzer(
|
|||||||
void ClangCompletionContextAnalyzer::analyze()
|
void ClangCompletionContextAnalyzer::analyze()
|
||||||
{
|
{
|
||||||
QTC_ASSERT(m_interface, return);
|
QTC_ASSERT(m_interface, return);
|
||||||
const int startOfName = findStartOfName();
|
|
||||||
m_positionForProposal = startOfName;
|
|
||||||
setActionAndClangPosition(PassThroughToLibClang, -1);
|
setActionAndClangPosition(PassThroughToLibClang, -1);
|
||||||
|
|
||||||
const int endOfOperator = skipPrecedingWhitespace(startOfName);
|
ActivationSequenceContextProcessor activationSequenceContextProcessor(m_interface);
|
||||||
m_completionOperator = T_EOF_SYMBOL;
|
m_completionOperator = activationSequenceContextProcessor.completionKind();
|
||||||
m_positionEndOfExpression = startOfOperator(endOfOperator, &m_completionOperator,
|
int afterOperatorPosition = activationSequenceContextProcessor.positionAfterOperator();
|
||||||
/*want function call =*/ true);
|
m_positionEndOfExpression = activationSequenceContextProcessor.positionBeforeOperator();
|
||||||
|
m_positionForProposal = activationSequenceContextProcessor.positionAfterOperator();
|
||||||
|
|
||||||
if (isTokenForPassThrough(m_completionOperator)) {
|
const bool actionIsSet = handleNonFunctionCall(afterOperatorPosition);
|
||||||
setActionAndClangPosition(PassThroughToLibClang, endOfOperator);
|
|
||||||
return;
|
if (!actionIsSet) {
|
||||||
} else if (m_completionOperator == T_DOXY_COMMENT) {
|
handleCommaInFunctionCall();
|
||||||
setActionAndClangPosition(CompleteDoxygenKeyword, -1);
|
handleFunctionCall(afterOperatorPosition);
|
||||||
return;
|
|
||||||
} else if (m_completionOperator == T_POUND) {
|
|
||||||
// TODO: Check if libclang can complete preprocessor directives
|
|
||||||
setActionAndClangPosition(CompletePreprocessorDirective, -1);
|
|
||||||
return;
|
|
||||||
} else if (isTokenForIncludePathCompletion(m_completionOperator)) {
|
|
||||||
setActionAndClangPosition(CompleteIncludePath, -1);
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ExpressionUnderCursor expressionUnderCursor(m_languageFeatures);
|
|
||||||
QTextCursor textCursor(m_interface->textDocument());
|
|
||||||
|
|
||||||
if (m_completionOperator == T_COMMA) { // For function hints
|
|
||||||
textCursor.setPosition(m_positionEndOfExpression);
|
|
||||||
const int start = expressionUnderCursor.startOfFunctionCall(textCursor);
|
|
||||||
QTC_ASSERT(start != -1, setActionAndClangPosition(PassThroughToLibClang, startOfName); return);
|
|
||||||
m_positionEndOfExpression = start;
|
|
||||||
m_positionForProposal = start + 1; // After '(' of function call
|
|
||||||
m_completionOperator = T_LPAREN;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (m_completionOperator == T_LPAREN) {
|
|
||||||
textCursor.setPosition(m_positionEndOfExpression);
|
|
||||||
const QString expression = expressionUnderCursor(textCursor);
|
|
||||||
|
|
||||||
if (expression.endsWith(QLatin1String("SIGNAL"))) {
|
|
||||||
setActionAndClangPosition(CompleteSignal, endOfOperator);
|
|
||||||
} else if (expression.endsWith(QLatin1String("SLOT"))) {
|
|
||||||
setActionAndClangPosition(CompleteSlot, endOfOperator);
|
|
||||||
} else if (m_interface->position() != endOfOperator) {
|
|
||||||
// No function completion if cursor is not after '(' or ','
|
|
||||||
m_positionForProposal = startOfName;
|
|
||||||
setActionAndClangPosition(PassThroughToLibClang, endOfOperator);
|
|
||||||
} else {
|
|
||||||
const FunctionInfo functionInfo = analyzeFunctionCall(endOfOperator);
|
|
||||||
m_functionName = functionInfo.functionName;
|
|
||||||
setActionAndClangPosition(PassThroughToLibClangAfterLeftParen,
|
|
||||||
functionInfo.functionNamePosition);
|
|
||||||
}
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
QTC_CHECK(!"Unexpected completion context");
|
|
||||||
setActionAndClangPosition(PassThroughToLibClang, startOfName);
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ClangCompletionContextAnalyzer::FunctionInfo ClangCompletionContextAnalyzer::analyzeFunctionCall(
|
ClangCompletionContextAnalyzer::FunctionInfo ClangCompletionContextAnalyzer::analyzeFunctionCall(
|
||||||
@@ -260,104 +142,6 @@ int ClangCompletionContextAnalyzer::skipPrecedingWhitespace(int position) const
|
|||||||
return position;
|
return position;
|
||||||
}
|
}
|
||||||
|
|
||||||
int ClangCompletionContextAnalyzer::startOfOperator(int pos,
|
|
||||||
unsigned *kind,
|
|
||||||
bool wantFunctionCall) const
|
|
||||||
{
|
|
||||||
const QChar ch = pos > -1 ? m_interface->characterAt(pos - 1) : QChar();
|
|
||||||
const QChar ch2 = pos > 0 ? m_interface->characterAt(pos - 2) : QChar();
|
|
||||||
const QChar ch3 = pos > 1 ? m_interface->characterAt(pos - 3) : QChar();
|
|
||||||
|
|
||||||
int start = pos - activationSequenceChar(ch, ch2, ch3, kind, wantFunctionCall);
|
|
||||||
if (start != pos) {
|
|
||||||
QTextCursor tc(m_interface->textDocument());
|
|
||||||
tc.setPosition(pos);
|
|
||||||
|
|
||||||
// Include completion: make sure the quote character is the first one on the line
|
|
||||||
if (*kind == T_STRING_LITERAL) {
|
|
||||||
QTextCursor s = tc;
|
|
||||||
s.movePosition(QTextCursor::StartOfLine, QTextCursor::KeepAnchor);
|
|
||||||
QString sel = s.selectedText();
|
|
||||||
if (sel.indexOf(QLatin1Char('"')) < sel.length() - 1) {
|
|
||||||
*kind = T_EOF_SYMBOL;
|
|
||||||
start = pos;
|
|
||||||
}
|
|
||||||
} else if (*kind == T_COMMA) {
|
|
||||||
ExpressionUnderCursor expressionUnderCursor(m_languageFeatures);
|
|
||||||
if (expressionUnderCursor.startOfFunctionCall(tc) == -1) {
|
|
||||||
*kind = T_EOF_SYMBOL;
|
|
||||||
start = pos;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
SimpleLexer tokenize;
|
|
||||||
tokenize.setLanguageFeatures(m_languageFeatures);
|
|
||||||
tokenize.setSkipComments(false);
|
|
||||||
const Tokens &tokens = tokenize(tc.block().text(), BackwardsScanner::previousBlockState(tc.block()));
|
|
||||||
const int tokenIdx = SimpleLexer::tokenBefore(tokens, qMax(0, tc.positionInBlock() - 1)); // get the token at the left of the cursor
|
|
||||||
const Token tk = (tokenIdx == -1) ? Token() : tokens.at(tokenIdx);
|
|
||||||
|
|
||||||
if (*kind == T_DOXY_COMMENT && !(tk.is(T_DOXY_COMMENT) || tk.is(T_CPP_DOXY_COMMENT))) {
|
|
||||||
*kind = T_EOF_SYMBOL;
|
|
||||||
start = pos;
|
|
||||||
}
|
|
||||||
// Don't complete in comments or strings, but still check for include completion
|
|
||||||
else if (tk.is(T_COMMENT) || tk.is(T_CPP_COMMENT) ||
|
|
||||||
(tk.isLiteral() && (*kind != T_STRING_LITERAL
|
|
||||||
&& *kind != T_ANGLE_STRING_LITERAL
|
|
||||||
&& *kind != T_SLASH))) {
|
|
||||||
*kind = T_EOF_SYMBOL;
|
|
||||||
start = pos;
|
|
||||||
}
|
|
||||||
// Include completion: can be triggered by slash, but only in a string
|
|
||||||
else if (*kind == T_SLASH && (tk.isNot(T_STRING_LITERAL) && tk.isNot(T_ANGLE_STRING_LITERAL))) {
|
|
||||||
*kind = T_EOF_SYMBOL;
|
|
||||||
start = pos;
|
|
||||||
}
|
|
||||||
else if (*kind == T_LPAREN) {
|
|
||||||
if (tokenIdx > 0) {
|
|
||||||
const Token &previousToken = tokens.at(tokenIdx - 1); // look at the token at the left of T_LPAREN
|
|
||||||
switch (previousToken.kind()) {
|
|
||||||
case T_IDENTIFIER:
|
|
||||||
case T_GREATER:
|
|
||||||
case T_SIGNAL:
|
|
||||||
case T_SLOT:
|
|
||||||
break; // good
|
|
||||||
|
|
||||||
default:
|
|
||||||
// that's a bad token :)
|
|
||||||
*kind = T_EOF_SYMBOL;
|
|
||||||
start = pos;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Check for include preprocessor directive
|
|
||||||
else if (*kind == T_STRING_LITERAL || *kind == T_ANGLE_STRING_LITERAL || *kind == T_SLASH) {
|
|
||||||
bool include = false;
|
|
||||||
if (tokens.size() >= 3) {
|
|
||||||
if (tokens.at(0).is(T_POUND) && tokens.at(1).is(T_IDENTIFIER) && (tokens.at(2).is(T_STRING_LITERAL) ||
|
|
||||||
tokens.at(2).is(T_ANGLE_STRING_LITERAL))) {
|
|
||||||
const Token &directiveToken = tokens.at(1);
|
|
||||||
QString directive = tc.block().text().mid(directiveToken.bytesBegin(),
|
|
||||||
directiveToken.bytes());
|
|
||||||
if (directive == QLatin1String("include") ||
|
|
||||||
directive == QLatin1String("include_next") ||
|
|
||||||
directive == QLatin1String("import")) {
|
|
||||||
include = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!include) {
|
|
||||||
*kind = T_EOF_SYMBOL;
|
|
||||||
start = pos;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return start;
|
|
||||||
}
|
|
||||||
|
|
||||||
void ClangCompletionContextAnalyzer::setActionAndClangPosition(CompletionAction action,
|
void ClangCompletionContextAnalyzer::setActionAndClangPosition(CompletionAction action,
|
||||||
int position)
|
int position)
|
||||||
{
|
{
|
||||||
@@ -366,5 +150,68 @@ void ClangCompletionContextAnalyzer::setActionAndClangPosition(CompletionAction
|
|||||||
m_positionForClang = position;
|
m_positionForClang = position;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ClangCompletionContextAnalyzer::setAction(ClangCompletionContextAnalyzer::CompletionAction action)
|
||||||
|
{
|
||||||
|
setActionAndClangPosition(action, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ClangCompletionContextAnalyzer::handleCommaInFunctionCall()
|
||||||
|
{
|
||||||
|
if (m_completionOperator == T_COMMA) {
|
||||||
|
ExpressionUnderCursor expressionUnderCursor(m_languageFeatures);
|
||||||
|
QTextCursor textCursor(m_interface->textDocument());
|
||||||
|
textCursor.setPosition(m_positionEndOfExpression);
|
||||||
|
const int start = expressionUnderCursor.startOfFunctionCall(textCursor);
|
||||||
|
m_positionEndOfExpression = start;
|
||||||
|
m_positionForProposal = start + 1; // After '(' of function call
|
||||||
|
m_completionOperator = T_LPAREN;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ClangCompletionContextAnalyzer::handleFunctionCall(int afterOperatorPosition)
|
||||||
|
{
|
||||||
|
if (m_completionOperator == T_LPAREN) {
|
||||||
|
ExpressionUnderCursor expressionUnderCursor(m_languageFeatures);
|
||||||
|
QTextCursor textCursor(m_interface->textDocument());
|
||||||
|
textCursor.setPosition(m_positionEndOfExpression);
|
||||||
|
const QString expression = expressionUnderCursor(textCursor);
|
||||||
|
|
||||||
|
if (expression.endsWith(QLatin1String("SIGNAL"))) {
|
||||||
|
setActionAndClangPosition(CompleteSignal, afterOperatorPosition);
|
||||||
|
} else if (expression.endsWith(QLatin1String("SLOT"))) {
|
||||||
|
setActionAndClangPosition(CompleteSlot, afterOperatorPosition);
|
||||||
|
} else if (m_interface->position() != afterOperatorPosition) {
|
||||||
|
// No function completion if cursor is not after '(' or ','
|
||||||
|
m_positionForProposal = afterOperatorPosition;
|
||||||
|
setActionAndClangPosition(PassThroughToLibClang, afterOperatorPosition);
|
||||||
|
} else {
|
||||||
|
const FunctionInfo functionInfo = analyzeFunctionCall(afterOperatorPosition);
|
||||||
|
m_functionName = functionInfo.functionName;
|
||||||
|
setActionAndClangPosition(PassThroughToLibClangAfterLeftParen,
|
||||||
|
functionInfo.functionNamePosition);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ClangCompletionContextAnalyzer::handleNonFunctionCall(int position)
|
||||||
|
{
|
||||||
|
if (isTokenForPassThrough(m_completionOperator)) {
|
||||||
|
setActionAndClangPosition(PassThroughToLibClang, position);
|
||||||
|
return true;
|
||||||
|
} else if (m_completionOperator == T_DOXY_COMMENT) {
|
||||||
|
setAction(CompleteDoxygenKeyword);
|
||||||
|
return true;
|
||||||
|
} else if (m_completionOperator == T_POUND) {
|
||||||
|
// TODO: Check if libclang can complete preprocessor directives
|
||||||
|
setAction(CompletePreprocessorDirective);
|
||||||
|
return true;
|
||||||
|
} else if (isTokenForIncludePathCompletion(m_completionOperator)) {
|
||||||
|
setAction(CompleteIncludePath);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace Internal
|
} // namespace Internal
|
||||||
} // namespace ClangCodeModel
|
} // namespace ClangCodeModel
|
||||||
|
|||||||
@@ -40,10 +40,12 @@ namespace TextEditor { class AssistInterface; }
|
|||||||
namespace ClangCodeModel {
|
namespace ClangCodeModel {
|
||||||
namespace Internal {
|
namespace Internal {
|
||||||
|
|
||||||
|
class ClangCompletionAssistInterface;
|
||||||
|
|
||||||
class ClangCompletionContextAnalyzer
|
class ClangCompletionContextAnalyzer
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
ClangCompletionContextAnalyzer(const TextEditor::AssistInterface *assistInterface,
|
ClangCompletionContextAnalyzer(const ClangCompletionAssistInterface *assistInterface,
|
||||||
CPlusPlus::LanguageFeatures languageFeatures);
|
CPlusPlus::LanguageFeatures languageFeatures);
|
||||||
void analyze();
|
void analyze();
|
||||||
|
|
||||||
@@ -72,16 +74,21 @@ private:
|
|||||||
|
|
||||||
int findStartOfName(int position = -1) const;
|
int findStartOfName(int position = -1) const;
|
||||||
int skipPrecedingWhitespace(int position) const;
|
int skipPrecedingWhitespace(int position) const;
|
||||||
int startOfOperator(int position, unsigned *kind, bool wantFunctionCall) const;
|
|
||||||
|
|
||||||
void setActionAndClangPosition(CompletionAction action, int position);
|
void setActionAndClangPosition(CompletionAction action, int position);
|
||||||
|
void setAction(CompletionAction action);
|
||||||
|
|
||||||
const TextEditor::AssistInterface * const m_interface; // Not owned
|
bool handleNonFunctionCall(int position);
|
||||||
|
void handleCommaInFunctionCall();
|
||||||
|
void handleFunctionCall(int endOfOperator);
|
||||||
|
|
||||||
|
private:
|
||||||
|
const ClangCompletionAssistInterface *m_interface; // Not owned
|
||||||
const CPlusPlus::LanguageFeatures m_languageFeatures; // TODO: Get from assistInterface?!
|
const CPlusPlus::LanguageFeatures m_languageFeatures; // TODO: Get from assistInterface?!
|
||||||
|
|
||||||
// Results
|
// Results
|
||||||
CompletionAction m_completionAction = PassThroughToLibClang;
|
CompletionAction m_completionAction = PassThroughToLibClang;
|
||||||
unsigned m_completionOperator = CPlusPlus::T_EOF_SYMBOL;
|
CPlusPlus::Kind m_completionOperator = CPlusPlus::T_EOF_SYMBOL;
|
||||||
int m_positionForProposal = -1;
|
int m_positionForProposal = -1;
|
||||||
int m_positionForClang = -1;
|
int m_positionForClang = -1;
|
||||||
int m_positionEndOfExpression = -1;
|
int m_positionEndOfExpression = -1;
|
||||||
|
|||||||
@@ -1,234 +0,0 @@
|
|||||||
/****************************************************************************
|
|
||||||
**
|
|
||||||
** Copyright (C) 2015 The Qt Company Ltd.
|
|
||||||
** Contact: http://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 http://www.qt.io/terms-conditions. For further information
|
|
||||||
** use the contact form at http://www.qt.io/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 or version 3 as published by the Free
|
|
||||||
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
|
||||||
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
|
||||||
** following information to ensure the GNU Lesser General Public License
|
|
||||||
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
|
||||||
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
||||||
**
|
|
||||||
** In addition, as a special exception, The Qt Company gives you certain additional
|
|
||||||
** rights. These rights are described in The Qt Company LGPL Exception
|
|
||||||
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
||||||
**
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
#include "clangcompletioncontextanalyzertest.h"
|
|
||||||
|
|
||||||
#include <clangcodemodel/clangcompletioncontextanalyzer.h>
|
|
||||||
#include <texteditor/codeassist/assistinterface.h>
|
|
||||||
|
|
||||||
#include <utils/qtcassert.h>
|
|
||||||
|
|
||||||
#include <QDebug>
|
|
||||||
#include <QTest>
|
|
||||||
#include <QTextDocument>
|
|
||||||
|
|
||||||
using namespace CPlusPlus;
|
|
||||||
using namespace ClangCodeModel;
|
|
||||||
using namespace ClangCodeModel::Internal;
|
|
||||||
using namespace ClangCodeModel::Internal::Tests;
|
|
||||||
|
|
||||||
Q_DECLARE_METATYPE(ClangCodeModel::Internal::ClangCompletionContextAnalyzer::CompletionAction)
|
|
||||||
|
|
||||||
QT_BEGIN_NAMESPACE
|
|
||||||
namespace QTest {
|
|
||||||
|
|
||||||
template<> char *toString(const ClangCompletionContextAnalyzer::CompletionAction &action)
|
|
||||||
{
|
|
||||||
using CCA = ClangCompletionContextAnalyzer;
|
|
||||||
|
|
||||||
switch (action) {
|
|
||||||
case CCA::PassThroughToLibClang:
|
|
||||||
return qstrdup("PassThroughToLibClang");
|
|
||||||
case CCA::PassThroughToLibClangAfterLeftParen:
|
|
||||||
return qstrdup("PassThroughToLibClangAfterLeftParen");
|
|
||||||
case CCA::CompleteDoxygenKeyword:
|
|
||||||
return qstrdup("CompleteDoxygenKeyword");
|
|
||||||
case CCA::CompleteIncludePath:
|
|
||||||
return qstrdup("CompleteIncludePath");
|
|
||||||
case CCA::CompletePreprocessorDirective:
|
|
||||||
return qstrdup("CompletePreprocessorDirective");
|
|
||||||
case CCA::CompleteSignal:
|
|
||||||
return qstrdup("CompleteSignal");
|
|
||||||
case CCA::CompleteSlot:
|
|
||||||
return qstrdup("CompleteSlot");
|
|
||||||
}
|
|
||||||
return qstrdup("Unexpected Value");
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace QTest
|
|
||||||
QT_END_NAMESPACE
|
|
||||||
|
|
||||||
namespace {
|
|
||||||
|
|
||||||
typedef QByteArray _;
|
|
||||||
|
|
||||||
class DummyAssistInterface : public TextEditor::AssistInterface
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
DummyAssistInterface(const QByteArray &text, int position)
|
|
||||||
: AssistInterface(new QTextDocument(QString::fromUtf8(text)),
|
|
||||||
position,
|
|
||||||
QLatin1String("<testdocument>"),
|
|
||||||
TextEditor::ActivationCharacter)
|
|
||||||
{}
|
|
||||||
~DummyAssistInterface() { delete textDocument(); }
|
|
||||||
};
|
|
||||||
|
|
||||||
class TestDocument
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
TestDocument(const QByteArray &theSource)
|
|
||||||
: source(theSource)
|
|
||||||
, position(theSource.lastIndexOf('@')) // Use 'lastIndexOf' due to doxygen: "//! @keyword"
|
|
||||||
{
|
|
||||||
QTC_CHECK(position != -1);
|
|
||||||
source.remove(position, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
QByteArray source;
|
|
||||||
int position;
|
|
||||||
};
|
|
||||||
|
|
||||||
bool isAPassThroughToLibClangAction(ClangCompletionContextAnalyzer::CompletionAction action)
|
|
||||||
{
|
|
||||||
return action == ClangCompletionContextAnalyzer::PassThroughToLibClang
|
|
||||||
|| action == ClangCompletionContextAnalyzer::PassThroughToLibClangAfterLeftParen;
|
|
||||||
}
|
|
||||||
|
|
||||||
ClangCompletionContextAnalyzer runAnalyzer(const TestDocument &testDocument)
|
|
||||||
{
|
|
||||||
DummyAssistInterface assistInterface(testDocument.source, testDocument.position);
|
|
||||||
ClangCompletionContextAnalyzer analyzer(&assistInterface, LanguageFeatures::defaultFeatures());
|
|
||||||
analyzer.analyze();
|
|
||||||
return analyzer;
|
|
||||||
}
|
|
||||||
|
|
||||||
} // anonymous namespace
|
|
||||||
|
|
||||||
void ClangCompletionContextAnalyzerTest::testPassThroughToClangAndSignalSlotRecognition_data()
|
|
||||||
{
|
|
||||||
QTest::addColumn<QByteArray>("givenSource");
|
|
||||||
QTest::addColumn<ClangCompletionContextAnalyzer::CompletionAction>("expectedCompletionAction");
|
|
||||||
QTest::addColumn<int>("expectedDiffBetweenCursorAndCalculatedClangPosition");
|
|
||||||
QTest::addColumn<int>("expectedDiffBetweenCursorAndCalculatedProposalPosition");
|
|
||||||
|
|
||||||
using CCA = ClangCompletionContextAnalyzer;
|
|
||||||
|
|
||||||
QTest::newRow("members - dot 1") << _("o.mem@") << CCA::PassThroughToLibClang << -3 << -3;
|
|
||||||
QTest::newRow("members - dot 2") << _("o. mem@") << CCA::PassThroughToLibClang << -4 << -3;
|
|
||||||
QTest::newRow("members - dot 3") << _("o.@mem") << CCA::PassThroughToLibClang << 0 << 0;
|
|
||||||
QTest::newRow("members - dot 4") << _("o. @ mem") << CCA::PassThroughToLibClang << -1 << 0;
|
|
||||||
QTest::newRow("members - arrow 1") << _("o->mem@") << CCA::PassThroughToLibClang << -3 << -3;
|
|
||||||
QTest::newRow("members - arrow 2") << _("o-> mem@") << CCA::PassThroughToLibClang << -4 << -3;
|
|
||||||
QTest::newRow("members - arrow 3") << _("o->@mem") << CCA::PassThroughToLibClang << 0 << 0;
|
|
||||||
QTest::newRow("members - arrow 4") << _("o-> @ mem") << CCA::PassThroughToLibClang << -1 << 0;
|
|
||||||
|
|
||||||
QTest::newRow("call 1") << _("f(@") << CCA::PassThroughToLibClangAfterLeftParen << -2 << 0;
|
|
||||||
QTest::newRow("call 2") << _("f(1,@") << CCA::PassThroughToLibClangAfterLeftParen << -4 << -2;
|
|
||||||
QTest::newRow("call 3") << _("f(1, @") << CCA::PassThroughToLibClang << -1 << 0;
|
|
||||||
|
|
||||||
QTest::newRow("qt4 signals 1") << _("SIGNAL(@") << CCA::CompleteSignal << 0 << 0;
|
|
||||||
QTest::newRow("qt4 signals 2") << _("SIGNAL(foo@") << CCA::CompleteSignal << -3 << -3;
|
|
||||||
QTest::newRow("qt4 slots 1") << _("SLOT(@") << CCA::CompleteSlot << 0 << 0;
|
|
||||||
QTest::newRow("qt4 slots 2") << _("SLOT(foo@") << CCA::CompleteSlot << -3 << -3;
|
|
||||||
}
|
|
||||||
|
|
||||||
void ClangCompletionContextAnalyzerTest::testPassThroughToClangAndSignalSlotRecognition()
|
|
||||||
{
|
|
||||||
QFETCH(QByteArray, givenSource);
|
|
||||||
QFETCH(ClangCompletionContextAnalyzer::CompletionAction, expectedCompletionAction);
|
|
||||||
QFETCH(int, expectedDiffBetweenCursorAndCalculatedClangPosition);
|
|
||||||
QFETCH(int, expectedDiffBetweenCursorAndCalculatedProposalPosition);
|
|
||||||
|
|
||||||
const TestDocument testDocument(givenSource);
|
|
||||||
ClangCompletionContextAnalyzer analyzer = runAnalyzer(testDocument);
|
|
||||||
|
|
||||||
QCOMPARE(analyzer.completionAction(), expectedCompletionAction);
|
|
||||||
QCOMPARE(analyzer.positionForClang() - testDocument.position,
|
|
||||||
expectedDiffBetweenCursorAndCalculatedClangPosition);
|
|
||||||
QCOMPARE(analyzer.positionForProposal() - testDocument.position,
|
|
||||||
expectedDiffBetweenCursorAndCalculatedProposalPosition);
|
|
||||||
}
|
|
||||||
|
|
||||||
void ClangCompletionContextAnalyzerTest::testSpecialCompletionRecognition_data()
|
|
||||||
{
|
|
||||||
QTest::addColumn<QByteArray>("givenSource");
|
|
||||||
QTest::addColumn<ClangCompletionContextAnalyzer::CompletionAction>("expectedCompletionAction");
|
|
||||||
QTest::addColumn<int>("expectedDiffBetweenCursorAndCalculatedProposalPosition");
|
|
||||||
|
|
||||||
using CCA = ClangCompletionContextAnalyzer;
|
|
||||||
|
|
||||||
QTest::newRow("doxygen keywords 1") << _("//! \\@") << CCA::CompleteDoxygenKeyword << 0;
|
|
||||||
QTest::newRow("doxygen keywords 3") << _("//! @@") << CCA::CompleteDoxygenKeyword << 0;
|
|
||||||
QTest::newRow("doxygen keywords 2") << _("//! \\par@") << CCA::CompleteDoxygenKeyword << -3;
|
|
||||||
|
|
||||||
QTest::newRow("pp directives 1") << _("#@") << CCA::CompletePreprocessorDirective << 0;
|
|
||||||
QTest::newRow("pp directives 2") << _("#if@") << CCA::CompletePreprocessorDirective << -2;
|
|
||||||
|
|
||||||
QTest::newRow("pp include path 1") << _("#include \"foo@\"") << CCA::CompleteIncludePath << -3;
|
|
||||||
QTest::newRow("pp include path 2") << _("#include <foo@>") << CCA::CompleteIncludePath << -3;
|
|
||||||
QTest::newRow("pp include path 3") << _("#include <foo/@>") << CCA::CompleteIncludePath << 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
void ClangCompletionContextAnalyzerTest::testSpecialCompletionRecognition()
|
|
||||||
{
|
|
||||||
QFETCH(QByteArray, givenSource);
|
|
||||||
QFETCH(ClangCompletionContextAnalyzer::CompletionAction, expectedCompletionAction);
|
|
||||||
QFETCH(int, expectedDiffBetweenCursorAndCalculatedProposalPosition);
|
|
||||||
|
|
||||||
const TestDocument testDocument(givenSource);
|
|
||||||
ClangCompletionContextAnalyzer analyzer = runAnalyzer(testDocument);
|
|
||||||
|
|
||||||
QCOMPARE(analyzer.completionAction(), expectedCompletionAction);
|
|
||||||
QCOMPARE(analyzer.positionForClang(), -1);
|
|
||||||
QCOMPARE(analyzer.positionForProposal() - testDocument.position,
|
|
||||||
expectedDiffBetweenCursorAndCalculatedProposalPosition);
|
|
||||||
}
|
|
||||||
|
|
||||||
void ClangCompletionContextAnalyzerTest::testAvoidSpecialCompletionRecognition_data()
|
|
||||||
{
|
|
||||||
QTest::addColumn<QByteArray>("givenSource");
|
|
||||||
|
|
||||||
QTest::newRow("no special completion for literals 1") << _("\"@");
|
|
||||||
QTest::newRow("no special completion for literals 2") << _(" \"@");
|
|
||||||
QTest::newRow("no special completion for literals 3") << _("\"text\"@");
|
|
||||||
QTest::newRow("no special completion for literals 4") << _("\"hello cruel@ world\"");
|
|
||||||
QTest::newRow("no special completion for literals 5") << _("'@'");
|
|
||||||
QTest::newRow("no special completion for literals 6") << _("'a@'");
|
|
||||||
QTest::newRow("no special completion for comma operator") << _("a = b,@\"");
|
|
||||||
QTest::newRow("no special completion for doxygen marker not in doxygen comment 1") << _("@@");
|
|
||||||
QTest::newRow("no special completion for doxygen marker not in doxygen comment 2") << _("\\@");
|
|
||||||
QTest::newRow("no special completion in comments 1") << _("// text@");
|
|
||||||
QTest::newRow("no special completion in comments 2") << _("/* text@ */");
|
|
||||||
QTest::newRow("no special completion for slash") << _("5 /@");
|
|
||||||
QTest::newRow("no special completion for '(' 1") << _("(@");
|
|
||||||
QTest::newRow("no special completion for '(' 2") << _("((@");
|
|
||||||
QTest::newRow("no special completion for '(' 3") << _("*(@");
|
|
||||||
}
|
|
||||||
|
|
||||||
void ClangCompletionContextAnalyzerTest::testAvoidSpecialCompletionRecognition()
|
|
||||||
{
|
|
||||||
QFETCH(QByteArray, givenSource);
|
|
||||||
|
|
||||||
const TestDocument testDocument(givenSource);
|
|
||||||
ClangCompletionContextAnalyzer analyzer = runAnalyzer(testDocument);
|
|
||||||
|
|
||||||
QVERIFY(isAPassThroughToLibClangAction(analyzer.completionAction()));
|
|
||||||
}
|
|
||||||
@@ -28,32 +28,32 @@
|
|||||||
**
|
**
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
#ifndef CLANGCOMPLETIONCONTEXTANALYZERTEST_H
|
#ifndef CLANGCODEMODEL_INTERNAL_CLANGCOMPLETIONASSISTINTERFACE_H
|
||||||
#define CLANGCOMPLETIONCONTEXTANALYZERTEST_H
|
#define CLANGCODEMODEL_INTERNAL_CLANGCOMPLETIONASSISTINTERFACE_H
|
||||||
|
|
||||||
#include <QObject>
|
#include <texteditor/codeassist/assistinterface.h>
|
||||||
|
|
||||||
|
#include <cplusplus/Token.h>
|
||||||
|
|
||||||
namespace ClangCodeModel {
|
namespace ClangCodeModel {
|
||||||
namespace Internal {
|
namespace Internal {
|
||||||
namespace Tests {
|
|
||||||
|
|
||||||
class ClangCompletionContextAnalyzerTest : public QObject
|
class ClangCompletionAssistInterface: public TextEditor::AssistInterface
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
public:
|
||||||
|
ClangCompletionAssistInterface(const QByteArray &text,
|
||||||
|
int position)
|
||||||
|
: TextEditor::AssistInterface(text, position),
|
||||||
|
languageFeatures_(CPlusPlus::LanguageFeatures::defaultFeatures())
|
||||||
|
{}
|
||||||
|
|
||||||
private slots:
|
CPlusPlus::LanguageFeatures languageFeatures() const { return languageFeatures_; }
|
||||||
void testPassThroughToClangAndSignalSlotRecognition();
|
|
||||||
void testPassThroughToClangAndSignalSlotRecognition_data();
|
|
||||||
|
|
||||||
void testSpecialCompletionRecognition();
|
private:
|
||||||
void testSpecialCompletionRecognition_data();
|
CPlusPlus::LanguageFeatures languageFeatures_;
|
||||||
|
|
||||||
void testAvoidSpecialCompletionRecognition();
|
|
||||||
void testAvoidSpecialCompletionRecognition_data();
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace Tests
|
|
||||||
} // namespace Internal
|
} // namespace Internal
|
||||||
} // namespace ClangCodeModel
|
} // namespace ClangCodeModel
|
||||||
|
|
||||||
#endif // CLANGCOMPLETIONCONTEXTANALYZERTEST_H
|
#endif // CLANGCODEMODEL_INTERNAL_CLANGCOMPLETIONASSISTINTERFACE_H
|
||||||
52
tests/unit/mockup/texteditor/assistenums.h
Normal file
52
tests/unit/mockup/texteditor/assistenums.h
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
/****************************************************************************
|
||||||
|
**
|
||||||
|
** Copyright (C) 2015 The Qt Company Ltd.
|
||||||
|
** Contact: http://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 http://www.qt.io/terms-conditions. For further information
|
||||||
|
** use the contact form at http://www.qt.io/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 or version 3 as published by the Free
|
||||||
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
||||||
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
||||||
|
** following information to ensure the GNU Lesser General Public License
|
||||||
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
||||||
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
||||||
|
**
|
||||||
|
** In addition, as a special exception, The Qt Company gives you certain additional
|
||||||
|
** rights. These rights are described in The Qt Company LGPL Exception
|
||||||
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
||||||
|
**
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
#ifndef ASSISTENUMS_H
|
||||||
|
#define ASSISTENUMS_H
|
||||||
|
|
||||||
|
namespace TextEditor {
|
||||||
|
|
||||||
|
enum AssistKind
|
||||||
|
{
|
||||||
|
Completion,
|
||||||
|
QuickFix,
|
||||||
|
FollowSymbol
|
||||||
|
};
|
||||||
|
|
||||||
|
enum AssistReason
|
||||||
|
{
|
||||||
|
IdleEditor,
|
||||||
|
ActivationCharacter,
|
||||||
|
ExplicitlyInvoked
|
||||||
|
};
|
||||||
|
|
||||||
|
} // TextEditor
|
||||||
|
|
||||||
|
#endif // ASSISTENUMS_H
|
||||||
119
tests/unit/mockup/texteditor/codeassist/assistinterface.h
Normal file
119
tests/unit/mockup/texteditor/codeassist/assistinterface.h
Normal file
@@ -0,0 +1,119 @@
|
|||||||
|
/****************************************************************************
|
||||||
|
**
|
||||||
|
** Copyright (C) 2015 The Qt Company Ltd.
|
||||||
|
** Contact: http://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 http://www.qt.io/terms-conditions. For further information
|
||||||
|
** use the contact form at http://www.qt.io/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 or version 3 as published by the Free
|
||||||
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
||||||
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
||||||
|
** following information to ensure the GNU Lesser General Public License
|
||||||
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
||||||
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
||||||
|
**
|
||||||
|
** In addition, as a special exception, The Qt Company gives you certain additional
|
||||||
|
** rights. These rights are described in The Qt Company LGPL Exception
|
||||||
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
||||||
|
**
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
#ifndef ASSISTINTERFACE_H
|
||||||
|
#define ASSISTINTERFACE_H
|
||||||
|
|
||||||
|
#include <QTextCursor>
|
||||||
|
|
||||||
|
#include <QTextDocument>
|
||||||
|
|
||||||
|
#include "../assistenums.h"
|
||||||
|
|
||||||
|
namespace TextEditor {
|
||||||
|
|
||||||
|
class AssistInterface
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
AssistInterface(const QByteArray &text,
|
||||||
|
int position)
|
||||||
|
: textDocument_(QString::fromUtf8(text)),
|
||||||
|
position_(position)
|
||||||
|
{}
|
||||||
|
|
||||||
|
AssistInterface(QTextDocument *textDocument,
|
||||||
|
int position,
|
||||||
|
const QString &fileName,
|
||||||
|
AssistReason reason)
|
||||||
|
: textDocument_(textDocument),
|
||||||
|
fileName_(fileName),
|
||||||
|
position_(position),
|
||||||
|
reason_(reason)
|
||||||
|
{}
|
||||||
|
|
||||||
|
QTextDocument *textDocument() const;
|
||||||
|
virtual int position() const;
|
||||||
|
virtual QChar characterAt(int position) const;
|
||||||
|
virtual QString textAt(int position, int length) const;
|
||||||
|
virtual QString fileName() const;
|
||||||
|
virtual AssistReason reason() const;
|
||||||
|
|
||||||
|
|
||||||
|
private:
|
||||||
|
mutable QTextDocument textDocument_;
|
||||||
|
QString fileName_;
|
||||||
|
int position_;
|
||||||
|
AssistReason reason_ = IdleEditor;
|
||||||
|
};
|
||||||
|
|
||||||
|
inline QTextDocument *AssistInterface::textDocument() const
|
||||||
|
{
|
||||||
|
return &textDocument_;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline int AssistInterface::position() const
|
||||||
|
{
|
||||||
|
return position_;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline QChar AssistInterface::characterAt(int position) const
|
||||||
|
{
|
||||||
|
return textDocument_.characterAt(position);
|
||||||
|
}
|
||||||
|
|
||||||
|
inline QString AssistInterface::textAt(int position, int length) const
|
||||||
|
{
|
||||||
|
QTextCursor textCursor(&textDocument_);
|
||||||
|
if (position < 0)
|
||||||
|
position = 0;
|
||||||
|
textCursor.movePosition(QTextCursor::End);
|
||||||
|
if (position + length > textCursor.position())
|
||||||
|
length = textCursor.position() - position;
|
||||||
|
|
||||||
|
textCursor.setPosition(position);
|
||||||
|
textCursor.setPosition(position + length, QTextCursor::KeepAnchor);
|
||||||
|
|
||||||
|
// selectedText() returns U+2029 (PARAGRAPH SEPARATOR) instead of newline
|
||||||
|
return textCursor.selectedText().replace(QChar::ParagraphSeparator, QLatin1Char('\n'));
|
||||||
|
}
|
||||||
|
|
||||||
|
inline AssistReason AssistInterface::reason() const
|
||||||
|
{
|
||||||
|
return reason_;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline QString AssistInterface::fileName() const
|
||||||
|
{
|
||||||
|
return fileName_;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // ASSISTINTERFACE_H
|
||||||
|
|
||||||
220
tests/unit/unittest/activationsequencecontextprocessortest.cpp
Normal file
220
tests/unit/unittest/activationsequencecontextprocessortest.cpp
Normal file
@@ -0,0 +1,220 @@
|
|||||||
|
/****************************************************************************
|
||||||
|
**
|
||||||
|
** Copyright (C) 2015 The Qt Company Ltd.
|
||||||
|
** Contact: http://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 http://www.qt.io/terms-conditions. For further information
|
||||||
|
** use the contact form at http://www.qt.io/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 or version 3 as published by the Free
|
||||||
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
||||||
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
||||||
|
** following information to ensure the GNU Lesser General Public License
|
||||||
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
||||||
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
||||||
|
**
|
||||||
|
** In addition, as a special exception, The Qt Company gives you certain additional
|
||||||
|
** rights. These rights are described in The Qt Company LGPL Exception
|
||||||
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
||||||
|
**
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
|
||||||
|
#include <activationsequencecontextprocessor.h>
|
||||||
|
|
||||||
|
#include <clangcodemodel/clangcompletionassistinterface.h>
|
||||||
|
|
||||||
|
#include <cplusplus/Token.h>
|
||||||
|
|
||||||
|
#include <QTextCursor>
|
||||||
|
#include <QTextDocument>
|
||||||
|
|
||||||
|
#include <gmock/gmock.h>
|
||||||
|
#include <gmock/gmock-matchers.h>
|
||||||
|
#include <gtest/gtest.h>
|
||||||
|
#include "gtest-qt-printing.h"
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
using ContextProcessor = ClangCodeModel::Internal::ActivationSequenceContextProcessor;
|
||||||
|
using TextEditor::AssistInterface;
|
||||||
|
using ClangCodeModel::Internal::ClangCompletionAssistInterface;
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, TextCursorPosition)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("foobar", 4);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.textCursor_forTestOnly().position(), 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, StringLiteral)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("auto foo = \"bar\"", 12);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_EOF_SYMBOL);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, EndOfStringLiteral)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("auto foo = \"bar\"", 16);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_EOF_SYMBOL);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, FunctionCallComma)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("f(x, ", 4);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_COMMA);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, NonFunctionCallComma)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("int x, ", 6);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_EOF_SYMBOL);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, DoxygenComment)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("//! @", 5);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_DOXY_COMMENT);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, NonDoxygenComment)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("// @", 4);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_EOF_SYMBOL);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, Comment)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("//", 2);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_EOF_SYMBOL);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, InsideALiteral)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("\"foo\"", 2);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_EOF_SYMBOL);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, ShlashInsideAString)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("\"foo/bar\"", 5);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_EOF_SYMBOL);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, ShlashOutsideAString)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("foo/bar", 4);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_EOF_SYMBOL);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, FunctionLeftParen)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("foo(", 4);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_LPAREN);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, TemplateFunctionLeftParen)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("foo<X>(", 7);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_LPAREN);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, ExpressionLeftParen)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("x * (", 5);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_EOF_SYMBOL);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, AngleInclude)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("#include <foo/bar>", 10);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_ANGLE_STRING_LITERAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, SlashInclude)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("#include <foo/bar>", 14);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_SLASH);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, QuoteInclude)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("#include \"foo/bar\"", 10);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_STRING_LITERAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, SlashInExlude)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("#exclude <foo/bar>", 14);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_EOF_SYMBOL);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, QuoteExclude)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("#exclude \"foo/bar\"", 10);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_EOF_SYMBOL);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, SkipeWhiteSpacesBeforeCursor)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("x-> ", 7);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_ARROW);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ActivationSequeneContextProcessor, SkipIdentifier)
|
||||||
|
{
|
||||||
|
ClangCompletionAssistInterface interface("x->foo_", 7);
|
||||||
|
ContextProcessor processor{&interface};
|
||||||
|
|
||||||
|
ASSERT_THAT(processor.completionKind(), CPlusPlus::T_ARROW);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -133,14 +133,14 @@ TEST(ActivationSequenceProcessor, ArrowStar)
|
|||||||
|
|
||||||
TEST(ActivationSequenceProcessor, DoxyGenCommentBackSlash)
|
TEST(ActivationSequenceProcessor, DoxyGenCommentBackSlash)
|
||||||
{
|
{
|
||||||
ActivationSequenceProcessor processor(QStringLiteral("\\ "), 3, true);
|
ActivationSequenceProcessor processor(QStringLiteral(" \\"), 3, true);
|
||||||
|
|
||||||
ASSERT_THAT(processor, HasResult(T_DOXY_COMMENT, 1, 2));
|
ASSERT_THAT(processor, HasResult(T_DOXY_COMMENT, 1, 2));
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST(ActivationSequenceProcessor, DoxyGenCommentAt)
|
TEST(ActivationSequenceProcessor, DoxyGenCommentAt)
|
||||||
{
|
{
|
||||||
ActivationSequenceProcessor processor(QStringLiteral("@ "), 2, true);
|
ActivationSequenceProcessor processor(QStringLiteral(" @"), 2, true);
|
||||||
|
|
||||||
ASSERT_THAT(processor, HasResult(T_DOXY_COMMENT, 1, 1));
|
ASSERT_THAT(processor, HasResult(T_DOXY_COMMENT, 1, 1));
|
||||||
}
|
}
|
||||||
@@ -182,7 +182,7 @@ TEST(ActivationSequenceProcessor, PositionIsOne)
|
|||||||
|
|
||||||
TEST(ActivationSequenceProcessor, PositionIsTwo)
|
TEST(ActivationSequenceProcessor, PositionIsTwo)
|
||||||
{
|
{
|
||||||
ActivationSequenceProcessor processor(QStringLiteral("@ x"), 2, true);
|
ActivationSequenceProcessor processor(QStringLiteral(" @x"), 2, true);
|
||||||
|
|
||||||
ASSERT_THAT(processor, HasResult(T_DOXY_COMMENT, 1, 1));
|
ASSERT_THAT(processor, HasResult(T_DOXY_COMMENT, 1, 1));
|
||||||
}
|
}
|
||||||
|
|||||||
435
tests/unit/unittest/clangcompletioncontextanalyzertest.cpp
Normal file
435
tests/unit/unittest/clangcompletioncontextanalyzertest.cpp
Normal file
@@ -0,0 +1,435 @@
|
|||||||
|
/****************************************************************************
|
||||||
|
**
|
||||||
|
** Copyright (C) 2015 The Qt Company Ltd.
|
||||||
|
** Contact: http://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 http://www.qt.io/terms-conditions. For further information
|
||||||
|
** use the contact form at http://www.qt.io/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 or version 3 as published by the Free
|
||||||
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
||||||
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
||||||
|
** following information to ensure the GNU Lesser General Public License
|
||||||
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
||||||
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
||||||
|
**
|
||||||
|
** In addition, as a special exception, The Qt Company gives you certain additional
|
||||||
|
** rights. These rights are described in The Qt Company LGPL Exception
|
||||||
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
||||||
|
**
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
|
||||||
|
#include <clangcompletioncontextanalyzer.h>
|
||||||
|
|
||||||
|
#include <clangcodemodel/clangcompletionassistinterface.h>
|
||||||
|
|
||||||
|
#include <utils/qtcassert.h>
|
||||||
|
|
||||||
|
#include <QTextDocument>
|
||||||
|
|
||||||
|
#include <gmock/gmock.h>
|
||||||
|
#include <gmock/gmock-matchers.h>
|
||||||
|
#include <gtest/gtest.h>
|
||||||
|
#include "gtest-qt-printing.h"
|
||||||
|
|
||||||
|
namespace ClangCodeModel {
|
||||||
|
namespace Internal {
|
||||||
|
void PrintTo(const ClangCompletionContextAnalyzer::CompletionAction &completionAction, ::std::ostream* os)
|
||||||
|
{
|
||||||
|
using CCA = ClangCompletionContextAnalyzer;
|
||||||
|
|
||||||
|
switch (completionAction) {
|
||||||
|
case CCA::PassThroughToLibClang: *os << "PassThroughToLibClang"; break;
|
||||||
|
case CCA::PassThroughToLibClangAfterLeftParen: *os << "PassThroughToLibClangAfterLeftParen"; break;
|
||||||
|
case CCA::CompleteDoxygenKeyword: *os << "CompleteDoxygenKeyword"; break;
|
||||||
|
case CCA::CompleteIncludePath: *os << "CompleteIncludePath"; break;
|
||||||
|
case CCA::CompletePreprocessorDirective: *os << "CompletePreprocessorDirective"; break;
|
||||||
|
case CCA::CompleteSignal: *os << "CompleteSignal"; break;
|
||||||
|
case CCA::CompleteSlot: *os << "CompleteSlot"; break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
using ClangCodeModel::Internal::ClangCompletionAssistInterface;
|
||||||
|
|
||||||
|
class TestDocument
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
TestDocument(const QByteArray &theSource)
|
||||||
|
: source(theSource),
|
||||||
|
position(theSource.lastIndexOf('@')) // Use 'lastIndexOf' due to doxygen: "//! @keyword"
|
||||||
|
{
|
||||||
|
source.remove(position, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
QByteArray source;
|
||||||
|
int position;
|
||||||
|
};
|
||||||
|
|
||||||
|
using ::testing::PrintToString;
|
||||||
|
|
||||||
|
MATCHER_P4(HasResult, completionAction, positionForClang, positionForProposal, positionInText,
|
||||||
|
std::string(negation ? "hasn't" : "has")
|
||||||
|
+ " result of completion action " + PrintToString(completionAction)
|
||||||
|
+ " and position for clang " + PrintToString(positionForClang)
|
||||||
|
+ " and position for proprosal " + PrintToString(positionForProposal))
|
||||||
|
{
|
||||||
|
int positionForClangDifference = arg.positionForClang() - positionInText;
|
||||||
|
int positionForProposalDifference = arg.positionForProposal() - positionInText;
|
||||||
|
if (arg.completionAction() != completionAction
|
||||||
|
|| positionForClangDifference != positionForClang
|
||||||
|
|| positionForProposalDifference != positionForProposal) {
|
||||||
|
*result_listener << "completion action is " << PrintToString(arg.completionAction())
|
||||||
|
<< " and position for clang is " << PrintToString(positionForClangDifference)
|
||||||
|
<< " and position for proprosal is " << PrintToString(positionForProposalDifference);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
MATCHER_P4(HasResultWithoutClangDifference, completionAction, positionForClang, positionForProposal, positionInText,
|
||||||
|
std::string(negation ? "hasn't" : "has")
|
||||||
|
+ " result of completion action " + PrintToString(completionAction)
|
||||||
|
+ " and position for clang " + PrintToString(positionForClang)
|
||||||
|
+ " and position for proprosal " + PrintToString(positionForProposal))
|
||||||
|
{
|
||||||
|
int positionForProposalDifference = arg.positionForProposal() - positionInText;
|
||||||
|
if (arg.completionAction() != completionAction
|
||||||
|
|| arg.positionForClang() != positionForClang
|
||||||
|
|| positionForProposalDifference != positionForProposal) {
|
||||||
|
*result_listener << "completion action is " << PrintToString(arg.completionAction())
|
||||||
|
<< " and position for clang is " << PrintToString(arg.positionForClang())
|
||||||
|
<< " and position for proprosal is " << PrintToString(positionForProposalDifference);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
using CCA = ClangCodeModel::Internal::ClangCompletionContextAnalyzer;
|
||||||
|
|
||||||
|
bool isPassThrough(CCA::CompletionAction completionAction)
|
||||||
|
{
|
||||||
|
return completionAction != CCA::PassThroughToLibClang
|
||||||
|
&& completionAction != CCA::PassThroughToLibClangAfterLeftParen;
|
||||||
|
}
|
||||||
|
|
||||||
|
MATCHER(IsPassThroughToClang, std::string(negation ? "isn't" : "is") + " passed through to Clang")
|
||||||
|
{
|
||||||
|
auto completionAction = arg.completionAction();
|
||||||
|
|
||||||
|
if (isPassThrough(completionAction)) {
|
||||||
|
*result_listener << "completion action is " << PrintToString(completionAction);
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
class ClangCompletionContextAnalyzer : public ::testing::Test
|
||||||
|
{
|
||||||
|
protected:
|
||||||
|
ClangCodeModel::Internal::ClangCompletionContextAnalyzer runAnalyzer(const char *text);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
int positionInText = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
ClangCodeModel::Internal::ClangCompletionContextAnalyzer ClangCompletionContextAnalyzer::runAnalyzer(const char *text)
|
||||||
|
{
|
||||||
|
const TestDocument testDocument(text);
|
||||||
|
ClangCompletionAssistInterface assistInterface(testDocument.source, testDocument.position);
|
||||||
|
ClangCodeModel::Internal::ClangCompletionContextAnalyzer analyzer(&assistInterface,
|
||||||
|
CPlusPlus::LanguageFeatures::defaultFeatures());
|
||||||
|
|
||||||
|
positionInText = testDocument.position;
|
||||||
|
|
||||||
|
analyzer.analyze();
|
||||||
|
|
||||||
|
return analyzer;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, AtEndOfDotMember)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("o.mem@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResult(CCA::PassThroughToLibClang, -3, -3, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, AtEndOfDotMemberWithSpaceInside)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("o. mem@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResult(CCA::PassThroughToLibClang, -4, -4, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, AtBeginOfDotMember)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("o.@mem");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResult(CCA::PassThroughToLibClang, 0, 0, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, AtBeginOfDotMemberWithSpaceInside)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("o. @mem");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResult(CCA::PassThroughToLibClang, -1, -1, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, AtEndOfArrow)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("o->mem@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResult(CCA::PassThroughToLibClang, -3, -3, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, AtEndOfArrowWithSpaceInside)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("o-> mem@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResult(CCA::PassThroughToLibClang, -4, -4, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, AtBeginOfArrow)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("o->@mem");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResult(CCA::PassThroughToLibClang, 0, 0, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, AtBeginOfArrowWithSpaceInside)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("o-> @mem");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResult(CCA::PassThroughToLibClang, -1, -1, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, ParameteOneAtCall)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("f(@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResult(CCA::PassThroughToLibClangAfterLeftParen, -2, 0, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, ParameteTwoAtCall)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("f(1,@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResult(CCA::PassThroughToLibClangAfterLeftParen, -4, -2, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, ParameteTwoWithSpaceAtCall)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("f(1, @");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResult(CCA::PassThroughToLibClang, -1, -1, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, ParameteOneAtSignal)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("SIGNAL(@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResult(CCA::CompleteSignal, 0, 0, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, ParameteOneWithLettersAtSignal)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("SIGNAL(foo@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResult(CCA::CompleteSignal, -3, -3, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, ParameteOneAtSlot)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("SLOT(@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResult(CCA::CompleteSlot, -0, 0, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, ParameteOneWithLettersAtSlot)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("SLOT(foo@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResult(CCA::CompleteSlot, -3, -3, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, DoxygenWithBackslash)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("//! \\@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResultWithoutClangDifference(CCA::CompleteDoxygenKeyword, -1, 0, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, DoxygenWithAt)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("//! @@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResultWithoutClangDifference(CCA::CompleteDoxygenKeyword, -1, 0, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, DoxygenWithParameter)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("//! \\par@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResultWithoutClangDifference(CCA::CompleteDoxygenKeyword, -1, -3, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, Preprocessor)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("#@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResultWithoutClangDifference(CCA::CompletePreprocessorDirective, -1, 0, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, PreprocessorIf)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("#if@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResultWithoutClangDifference(CCA::CompletePreprocessorDirective, -1, -2, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, LocalInclude)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("#include \"foo@\"");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResultWithoutClangDifference(CCA::CompleteIncludePath, -1, -3, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, GlobalInclude)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("#include <foo@>");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResultWithoutClangDifference(CCA::CompleteIncludePath, -1, -3, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, GlocalIncludeWithDirectory)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("#include <foo/@>");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, HasResultWithoutClangDifference(CCA::CompleteIncludePath, -1, 0, positionInText));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, AfterQuote)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("\"@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, IsPassThroughToClang());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, AfterSpaceQuote)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer(" \"@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, IsPassThroughToClang());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, AfterQuotedText)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("\"text\"@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, IsPassThroughToClang());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, InQuotedText)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("\"hello cruel@ world\"");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, IsPassThroughToClang());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, SingleQuote)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("'@'");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, IsPassThroughToClang());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, AfterLetterInSingleQuoted)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("'a@'");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, IsPassThroughToClang());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, CommaOperator)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("a = b,@\"");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, IsPassThroughToClang());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, DoxygenMarkerInNonDoxygenComment)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("@@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, IsPassThroughToClang());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, DoxygenMarkerInNonDoxygenComment2)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("\\@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, IsPassThroughToClang());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, OneLineComment)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("// text@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, IsPassThroughToClang());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, BeginEndComment)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("/* text@ */");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, IsPassThroughToClang());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, Slash)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("5 /@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, IsPassThroughToClang());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, LeftParen)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("(@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, IsPassThroughToClang());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, TwoLeftParen)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("((@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, IsPassThroughToClang());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ClangCompletionContextAnalyzer, AsteriskLeftParen)
|
||||||
|
{
|
||||||
|
auto analyzer = runAnalyzer("*(@");
|
||||||
|
|
||||||
|
ASSERT_THAT(analyzer, IsPassThroughToClang());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
INCLUDEPATH += $$PWD
|
INCLUDEPATH += $$PWD
|
||||||
|
|
||||||
SOURCES += $$PWD/../../../src/libs/3rdparty/cplusplus/Token.cpp
|
#include($$PWD/../../../src/libs/3rdparty/cplusplus/cplusplus.pri)
|
||||||
|
include($$PWD/../../../src/libs/utils/utils-lib.pri)
|
||||||
HEADERS += $$PWD/../../../src/libs/3rdparty/cplusplus/Token.h
|
include($$PWD/../../../src/libs/cplusplus/cplusplus-lib.pri)
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ GTEST_DIR = $$GMOCK_DIR/gtest
|
|||||||
requires(exists($$GMOCK_DIR))
|
requires(exists($$GMOCK_DIR))
|
||||||
!exists($$GMOCK_DIR):message("No gmock is found! To enabe unit tests set GMOCK_DIR")
|
!exists($$GMOCK_DIR):message("No gmock is found! To enabe unit tests set GMOCK_DIR")
|
||||||
|
|
||||||
|
INCLUDEPATH += ../mockup
|
||||||
INCLUDEPATH += $$GTEST_DIR $$GTEST_DIR/include $$GMOCK_DIR $$GMOCK_DIR/include
|
INCLUDEPATH += $$GTEST_DIR $$GTEST_DIR/include $$GMOCK_DIR $$GMOCK_DIR/include
|
||||||
|
|
||||||
include(../../../src/libs/sqlite/sqlite-lib.pri)
|
include(../../../src/libs/sqlite/sqlite-lib.pri)
|
||||||
@@ -47,7 +48,6 @@ SOURCES += main.cpp \
|
|||||||
clientserverinprocesstest.cpp \
|
clientserverinprocesstest.cpp \
|
||||||
clientserveroutsideprocess.cpp \
|
clientserveroutsideprocess.cpp \
|
||||||
codecompletiontest.cpp \
|
codecompletiontest.cpp \
|
||||||
../../../src/libs/utils/qtcassert.cpp \
|
|
||||||
clangstringtest.cpp \
|
clangstringtest.cpp \
|
||||||
translationunittest.cpp \
|
translationunittest.cpp \
|
||||||
clangcodecompleteresultstest.cpp \
|
clangcodecompleteresultstest.cpp \
|
||||||
@@ -58,12 +58,13 @@ SOURCES += main.cpp \
|
|||||||
translationunitstest.cpp \
|
translationunitstest.cpp \
|
||||||
completionchunkstotextconvertertest.cpp \
|
completionchunkstotextconvertertest.cpp \
|
||||||
lineprefixertest.cpp \
|
lineprefixertest.cpp \
|
||||||
activationsequenceprocessortest.cpp
|
activationsequenceprocessortest.cpp \
|
||||||
|
clangcompletioncontextanalyzertest.cpp \
|
||||||
|
activationsequencecontextprocessortest.cpp
|
||||||
|
|
||||||
HEADERS += \
|
HEADERS += \
|
||||||
gtest-qt-printing.h \
|
gtest-qt-printing.h \
|
||||||
spydummy.h \
|
spydummy.h \
|
||||||
../../../src/libs/utils/qtcassert.h \
|
|
||||||
mockipclient.h \
|
mockipclient.h \
|
||||||
mockipcserver.h
|
mockipcserver.h
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user