2013-12-10 14:37:32 +01:00
|
|
|
/****************************************************************************
|
|
|
|
**
|
2014-01-08 12:00:22 +01:00
|
|
|
** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
|
2013-12-10 14:37:32 +01:00
|
|
|
** Contact: http://www.qt-project.org/legal
|
|
|
|
**
|
|
|
|
** This file is part of Qt Creator.
|
|
|
|
**
|
|
|
|
** Commercial License Usage
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
** a written agreement between you and Digia. For licensing terms and
|
|
|
|
** conditions see http://qt.digia.com/licensing. For further information
|
|
|
|
** use the contact form at http://qt.digia.com/contact-us.
|
|
|
|
**
|
|
|
|
** GNU Lesser General Public License Usage
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
|
|
** General Public License version 2.1 as published by the Free Software
|
|
|
|
** Foundation and appearing in the file LICENSE.LGPL included in the
|
|
|
|
** packaging of this file. Please review the following information to
|
|
|
|
** ensure the GNU Lesser General Public License version 2.1 requirements
|
|
|
|
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
|
|
**
|
|
|
|
** In addition, as a special exception, Digia gives you certain additional
|
|
|
|
** rights. These rights are described in the Digia Qt LGPL Exception
|
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
**
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include "clangcompletion.h"
|
|
|
|
#include "clangutils.h"
|
|
|
|
#include "pchmanager.h"
|
|
|
|
|
|
|
|
#include <coreplugin/icore.h>
|
|
|
|
#include <coreplugin/idocument.h>
|
|
|
|
#include <coreplugin/mimedatabase.h>
|
|
|
|
|
|
|
|
#include <cplusplus/BackwardsScanner.h>
|
|
|
|
#include <cplusplus/ExpressionUnderCursor.h>
|
|
|
|
#include <cplusplus/Token.h>
|
|
|
|
#include <cplusplus/MatchingText.h>
|
|
|
|
|
|
|
|
#include <cppeditor/cppeditorconstants.h>
|
|
|
|
|
|
|
|
#include <cpptools/cppdoxygen.h>
|
|
|
|
#include <cpptools/cppmodelmanagerinterface.h>
|
2014-07-30 16:29:02 +02:00
|
|
|
#include <cpptools/cppworkingcopy.h>
|
2013-12-10 14:37:32 +01:00
|
|
|
|
|
|
|
#include <texteditor/basetexteditor.h>
|
|
|
|
#include <texteditor/convenience.h>
|
|
|
|
#include <texteditor/codeassist/basicproposalitemlistmodel.h>
|
|
|
|
#include <texteditor/codeassist/basicproposalitem.h>
|
|
|
|
#include <texteditor/codeassist/functionhintproposal.h>
|
|
|
|
#include <texteditor/codeassist/genericproposal.h>
|
|
|
|
#include <texteditor/codeassist/ifunctionhintproposalmodel.h>
|
|
|
|
#include <texteditor/texteditorsettings.h>
|
|
|
|
#include <texteditor/completionsettings.h>
|
|
|
|
|
2014-06-20 15:44:56 +04:00
|
|
|
#include <utils/algorithm.h>
|
|
|
|
|
2013-12-10 14:37:32 +01:00
|
|
|
#include <QCoreApplication>
|
|
|
|
#include <QDirIterator>
|
|
|
|
#include <QTextCursor>
|
|
|
|
#include <QTextDocument>
|
|
|
|
|
2014-03-06 14:42:01 -03:00
|
|
|
static const bool DebugTiming = qgetenv("QTC_CLANG_VERBOSE") == "1";
|
2013-12-10 14:37:32 +01:00
|
|
|
|
|
|
|
using namespace ClangCodeModel;
|
|
|
|
using namespace ClangCodeModel::Internal;
|
|
|
|
using namespace CPlusPlus;
|
|
|
|
using namespace CppTools;
|
|
|
|
using namespace TextEditor;
|
|
|
|
|
|
|
|
static const char SNIPPET_ICON_PATH[] = ":/texteditor/images/snippet.png";
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
static QList<CodeCompletionResult> unfilteredCompletion(const ClangCompletionAssistInterface* interface,
|
|
|
|
const QString &fileName,
|
|
|
|
unsigned line, unsigned column,
|
|
|
|
QByteArray modifiedInput = QByteArray(),
|
|
|
|
bool isSignalSlotCompletion = false)
|
|
|
|
{
|
|
|
|
ClangCompleter::Ptr wrapper = interface->clangWrapper();
|
|
|
|
QMutexLocker lock(wrapper->mutex());
|
|
|
|
//### TODO: check if we're cancelled after we managed to acquire the mutex
|
|
|
|
|
|
|
|
wrapper->setFileName(fileName);
|
|
|
|
wrapper->setOptions(interface->options());
|
|
|
|
wrapper->setSignalSlotCompletion(isSignalSlotCompletion);
|
|
|
|
UnsavedFiles unsavedFiles = interface->unsavedFiles();
|
|
|
|
if (!modifiedInput.isEmpty())
|
|
|
|
unsavedFiles.insert(fileName, modifiedInput);
|
|
|
|
|
|
|
|
QTime t;
|
|
|
|
if (DebugTiming) {
|
|
|
|
qDebug() << "Here we go with ClangCompletionAssistProcessor....";
|
|
|
|
t.start();
|
|
|
|
}
|
|
|
|
|
|
|
|
QList<CodeCompletionResult> result = wrapper->codeCompleteAt(line, column + 1, unsavedFiles);
|
2014-06-20 18:34:34 +04:00
|
|
|
::Utils::sort(result);
|
2013-12-10 14:37:32 +01:00
|
|
|
|
|
|
|
if (DebugTiming)
|
|
|
|
qDebug() << "... Completion done in" << t.elapsed() << "ms, with" << result.count() << "items.";
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // Anonymous
|
|
|
|
|
|
|
|
namespace ClangCodeModel {
|
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
// -----------------------------
|
|
|
|
// ClangCompletionAssistProvider
|
|
|
|
// -----------------------------
|
|
|
|
ClangCompletionAssistProvider::ClangCompletionAssistProvider()
|
|
|
|
: m_clangCompletionWrapper(new ClangCodeModel::ClangCompleter)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
IAssistProcessor *ClangCompletionAssistProvider::createProcessor() const
|
|
|
|
{
|
|
|
|
return new ClangCompletionAssistProcessor;
|
|
|
|
}
|
|
|
|
|
|
|
|
IAssistInterface *ClangCompletionAssistProvider::createAssistInterface(
|
2014-08-19 15:59:29 +02:00
|
|
|
ProjectExplorer::Project *project, const QString &filePath,
|
2014-05-14 11:34:45 -04:00
|
|
|
QTextDocument *document, bool isObjCEnabled, int position, AssistReason reason) const
|
2013-12-10 14:37:32 +01:00
|
|
|
{
|
|
|
|
Q_UNUSED(project);
|
2014-05-14 11:34:45 -04:00
|
|
|
Q_UNUSED(isObjCEnabled);
|
2013-12-10 14:37:32 +01:00
|
|
|
|
|
|
|
CppModelManagerInterface *modelManager = CppModelManagerInterface::instance();
|
2014-08-19 15:59:29 +02:00
|
|
|
QList<ProjectPart::Ptr> parts = modelManager->projectPart(filePath);
|
2013-12-10 14:37:32 +01:00
|
|
|
if (parts.isEmpty())
|
|
|
|
parts += modelManager->fallbackProjectPart();
|
2014-06-25 17:23:19 +02:00
|
|
|
ProjectPart::HeaderPaths headerPaths;
|
|
|
|
QStringList options;
|
2013-12-10 14:37:32 +01:00
|
|
|
PchInfo::Ptr pchInfo;
|
|
|
|
foreach (ProjectPart::Ptr part, parts) {
|
|
|
|
if (part.isNull())
|
|
|
|
continue;
|
2014-08-19 15:59:29 +02:00
|
|
|
options = ClangCodeModel::Utils::createClangOptions(part, filePath);
|
2014-02-04 15:53:32 +01:00
|
|
|
pchInfo = PchManager::instance()->pchInfo(part);
|
2013-12-10 14:37:32 +01:00
|
|
|
if (!pchInfo.isNull())
|
|
|
|
options.append(ClangCodeModel::Utils::createPCHInclusionOptions(pchInfo->fileName()));
|
2014-06-25 17:23:19 +02:00
|
|
|
headerPaths = part->headerPaths;
|
2013-12-10 14:37:32 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return new ClangCodeModel::ClangCompletionAssistInterface(
|
|
|
|
m_clangCompletionWrapper,
|
2014-08-19 15:59:29 +02:00
|
|
|
document, position, filePath, reason,
|
2014-06-25 17:23:19 +02:00
|
|
|
options, headerPaths, pchInfo);
|
2013-12-10 14:37:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// ------------------------
|
|
|
|
// ClangAssistProposalModel
|
|
|
|
// ------------------------
|
|
|
|
class ClangAssistProposalModel : public TextEditor::BasicProposalItemListModel
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
ClangAssistProposalModel()
|
|
|
|
: TextEditor::BasicProposalItemListModel()
|
|
|
|
, m_sortable(false)
|
|
|
|
, m_completionOperator(T_EOF_SYMBOL)
|
|
|
|
, m_replaceDotForArrow(false)
|
|
|
|
{}
|
|
|
|
|
|
|
|
virtual bool isSortable(const QString &prefix) const;
|
|
|
|
bool m_sortable;
|
|
|
|
unsigned m_completionOperator;
|
|
|
|
bool m_replaceDotForArrow;
|
|
|
|
};
|
|
|
|
|
|
|
|
// -------------------
|
|
|
|
// ClangAssistProposal
|
|
|
|
// -------------------
|
|
|
|
class ClangAssistProposal : public TextEditor::GenericProposal
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
ClangAssistProposal(int cursorPos, TextEditor::IGenericProposalModel *model)
|
|
|
|
: TextEditor::GenericProposal(cursorPos, model)
|
|
|
|
, m_replaceDotForArrow(static_cast<ClangAssistProposalModel *>(model)->m_replaceDotForArrow)
|
|
|
|
{}
|
|
|
|
|
|
|
|
virtual bool isCorrective() const { return m_replaceDotForArrow; }
|
|
|
|
virtual void makeCorrection(BaseTextEditor *editor)
|
|
|
|
{
|
|
|
|
editor->setCursorPosition(basePosition() - 1);
|
|
|
|
editor->replace(1, QLatin1String("->"));
|
|
|
|
moveBasePosition(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
bool m_replaceDotForArrow;
|
|
|
|
};
|
|
|
|
|
|
|
|
// ----------------------
|
|
|
|
// ClangFunctionHintModel
|
|
|
|
// ----------------------
|
|
|
|
class ClangFunctionHintModel : public TextEditor::IFunctionHintProposalModel
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
ClangFunctionHintModel(const QList<CodeCompletionResult> functionSymbols)
|
|
|
|
: m_functionSymbols(functionSymbols)
|
|
|
|
, m_currentArg(-1)
|
|
|
|
{}
|
|
|
|
|
|
|
|
virtual void reset() {}
|
|
|
|
virtual int size() const { return m_functionSymbols.size(); }
|
|
|
|
virtual QString text(int index) const;
|
|
|
|
virtual int activeArgument(const QString &prefix) const;
|
|
|
|
|
|
|
|
private:
|
|
|
|
QList<ClangCodeModel::CodeCompletionResult> m_functionSymbols;
|
|
|
|
mutable int m_currentArg;
|
|
|
|
};
|
|
|
|
|
|
|
|
QString ClangFunctionHintModel::text(int index) const
|
|
|
|
{
|
|
|
|
#if 0
|
|
|
|
// TODO: add the boldening to the result
|
|
|
|
Overview overview;
|
|
|
|
overview.setShowReturnTypes(true);
|
|
|
|
overview.setShowArgumentNames(true);
|
|
|
|
overview.setMarkedArgument(m_currentArg + 1);
|
|
|
|
Function *f = m_functionSymbols.at(index);
|
|
|
|
|
|
|
|
const QString prettyMethod = overview(f->type(), f->name());
|
|
|
|
const int begin = overview.markedArgumentBegin();
|
|
|
|
const int end = overview.markedArgumentEnd();
|
|
|
|
|
|
|
|
QString hintText;
|
|
|
|
hintText += Qt::escape(prettyMethod.left(begin));
|
|
|
|
hintText += "<b>";
|
|
|
|
hintText += Qt::escape(prettyMethod.mid(begin, end - begin));
|
|
|
|
hintText += "</b>";
|
|
|
|
hintText += Qt::escape(prettyMethod.mid(end));
|
|
|
|
return hintText;
|
|
|
|
#endif
|
|
|
|
return m_functionSymbols.at(index).hint();
|
|
|
|
}
|
|
|
|
|
|
|
|
int ClangFunctionHintModel::activeArgument(const QString &prefix) const
|
|
|
|
{
|
|
|
|
int argnr = 0;
|
|
|
|
int parcount = 0;
|
|
|
|
SimpleLexer tokenize;
|
|
|
|
QList<CPlusPlus::Token> tokens = tokenize(prefix);
|
|
|
|
for (int i = 0; i < tokens.count(); ++i) {
|
|
|
|
const CPlusPlus::Token &tk = tokens.at(i);
|
|
|
|
if (tk.is(T_LPAREN))
|
|
|
|
++parcount;
|
|
|
|
else if (tk.is(T_RPAREN))
|
|
|
|
--parcount;
|
|
|
|
else if (! parcount && tk.is(T_COMMA))
|
|
|
|
++argnr;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parcount < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (argnr != m_currentArg)
|
|
|
|
m_currentArg = argnr;
|
|
|
|
|
|
|
|
return argnr;
|
|
|
|
}
|
|
|
|
|
|
|
|
class ClangAssistProposalItem : public TextEditor::BasicProposalItem
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
ClangAssistProposalItem() {}
|
|
|
|
|
|
|
|
virtual bool prematurelyApplies(const QChar &c) const;
|
|
|
|
virtual void applyContextualContent(TextEditor::BaseTextEditor *editor,
|
|
|
|
int basePosition) const;
|
|
|
|
|
|
|
|
void keepCompletionOperator(unsigned compOp) { m_completionOperator = compOp; }
|
|
|
|
|
|
|
|
bool isOverloaded() const
|
|
|
|
{ return !m_overloads.isEmpty(); }
|
|
|
|
void addOverload(const CodeCompletionResult &ccr)
|
|
|
|
{ m_overloads.append(ccr); }
|
|
|
|
|
|
|
|
CodeCompletionResult originalItem() const
|
|
|
|
{
|
|
|
|
const QVariant &v = data();
|
|
|
|
if (v.canConvert<CodeCompletionResult>())
|
|
|
|
return v.value<CodeCompletionResult>();
|
|
|
|
else
|
|
|
|
return CodeCompletionResult();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isCodeCompletionResult() const
|
|
|
|
{ return data().canConvert<CodeCompletionResult>(); }
|
|
|
|
|
|
|
|
private:
|
|
|
|
unsigned m_completionOperator;
|
|
|
|
mutable QChar m_typedChar;
|
|
|
|
QList<CodeCompletionResult> m_overloads;
|
|
|
|
};
|
|
|
|
|
|
|
|
/// @return True, because clang always returns priorities for sorting
|
|
|
|
bool ClangAssistProposalModel::isSortable(const QString &prefix) const
|
|
|
|
{
|
|
|
|
Q_UNUSED(prefix)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace Internal
|
|
|
|
} // namespace ClangCodeModel
|
|
|
|
|
|
|
|
bool ClangAssistProposalItem::prematurelyApplies(const QChar &typedChar) const
|
|
|
|
{
|
|
|
|
bool ok = false;
|
|
|
|
|
|
|
|
if (m_completionOperator == T_SIGNAL || m_completionOperator == T_SLOT)
|
|
|
|
ok = QString::fromLatin1("(,").contains(typedChar);
|
|
|
|
else if (m_completionOperator == T_STRING_LITERAL || m_completionOperator == T_ANGLE_STRING_LITERAL)
|
|
|
|
ok = (typedChar == QLatin1Char('/')) && text().endsWith(QLatin1Char('/'));
|
|
|
|
else if (!isCodeCompletionResult())
|
|
|
|
ok = (typedChar == QLatin1Char('(')); /* && data().canConvert<CompleteFunctionDeclaration>()*/ //###
|
|
|
|
else if (originalItem().completionKind() == CodeCompletionResult::ObjCMessageCompletionKind)
|
|
|
|
ok = QString::fromLatin1(";.,").contains(typedChar);
|
|
|
|
else
|
|
|
|
ok = QString::fromLatin1(";.,:(").contains(typedChar);
|
|
|
|
|
|
|
|
if (ok)
|
|
|
|
m_typedChar = typedChar;
|
|
|
|
|
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ClangAssistProposalItem::applyContextualContent(TextEditor::BaseTextEditor *editor,
|
|
|
|
int basePosition) const
|
|
|
|
{
|
|
|
|
const CodeCompletionResult ccr = originalItem();
|
|
|
|
|
|
|
|
QString toInsert = text();
|
|
|
|
QString extraChars;
|
|
|
|
int extraLength = 0;
|
|
|
|
int cursorOffset = 0;
|
|
|
|
|
|
|
|
bool autoParenthesesEnabled = true;
|
|
|
|
if (m_completionOperator == T_SIGNAL || m_completionOperator == T_SLOT) {
|
|
|
|
extraChars += QLatin1Char(')');
|
|
|
|
if (m_typedChar == QLatin1Char('(')) // Eat the opening parenthesis
|
|
|
|
m_typedChar = QChar();
|
|
|
|
} else if (m_completionOperator == T_STRING_LITERAL || m_completionOperator == T_ANGLE_STRING_LITERAL) {
|
|
|
|
if (!toInsert.endsWith(QLatin1Char('/'))) {
|
|
|
|
extraChars += QLatin1Char((m_completionOperator == T_ANGLE_STRING_LITERAL) ? '>' : '"');
|
|
|
|
} else {
|
|
|
|
if (m_typedChar == QLatin1Char('/')) // Eat the slash
|
|
|
|
m_typedChar = QChar();
|
|
|
|
}
|
|
|
|
} else if (ccr.isValid()) {
|
|
|
|
const CompletionSettings &completionSettings =
|
|
|
|
TextEditorSettings::instance()->completionSettings();
|
|
|
|
const bool autoInsertBrackets = completionSettings.m_autoInsertBrackets;
|
|
|
|
|
|
|
|
if (autoInsertBrackets &&
|
|
|
|
(ccr.completionKind() == CodeCompletionResult::FunctionCompletionKind
|
|
|
|
|| ccr.completionKind() == CodeCompletionResult::DestructorCompletionKind
|
|
|
|
|| ccr.completionKind() == CodeCompletionResult::SignalCompletionKind
|
|
|
|
|| ccr.completionKind() == CodeCompletionResult::SlotCompletionKind)) {
|
|
|
|
// When the user typed the opening parenthesis, he'll likely also type the closing one,
|
|
|
|
// in which case it would be annoying if we put the cursor after the already automatically
|
|
|
|
// inserted closing parenthesis.
|
|
|
|
const bool skipClosingParenthesis = m_typedChar != QLatin1Char('(');
|
|
|
|
|
|
|
|
if (completionSettings.m_spaceAfterFunctionName)
|
|
|
|
extraChars += QLatin1Char(' ');
|
|
|
|
extraChars += QLatin1Char('(');
|
|
|
|
if (m_typedChar == QLatin1Char('('))
|
|
|
|
m_typedChar = QChar();
|
|
|
|
|
|
|
|
// If the function doesn't return anything, automatically place the semicolon,
|
|
|
|
// unless we're doing a scope completion (then it might be function definition).
|
|
|
|
const QChar characterAtCursor = editor->textDocument()->characterAt(editor->position());
|
|
|
|
bool endWithSemicolon = m_typedChar == QLatin1Char(';')/*
|
|
|
|
|| (function->returnType()->isVoidType() && m_completionOperator != T_COLON_COLON)*/; //###
|
|
|
|
const QChar semicolon = m_typedChar.isNull() ? QLatin1Char(';') : m_typedChar;
|
|
|
|
|
|
|
|
if (endWithSemicolon && characterAtCursor == semicolon) {
|
|
|
|
endWithSemicolon = false;
|
|
|
|
m_typedChar = QChar();
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the function takes no arguments, automatically place the closing parenthesis
|
|
|
|
if (!isOverloaded() && !ccr.hasParameters() && skipClosingParenthesis) {
|
|
|
|
extraChars += QLatin1Char(')');
|
|
|
|
if (endWithSemicolon) {
|
|
|
|
extraChars += semicolon;
|
|
|
|
m_typedChar = QChar();
|
|
|
|
}
|
|
|
|
} else if (autoParenthesesEnabled) {
|
|
|
|
const QChar lookAhead = editor->textDocument()->characterAt(editor->position() + 1);
|
|
|
|
if (MatchingText::shouldInsertMatchingText(lookAhead)) {
|
|
|
|
extraChars += QLatin1Char(')');
|
|
|
|
--cursorOffset;
|
|
|
|
if (endWithSemicolon) {
|
|
|
|
extraChars += semicolon;
|
|
|
|
--cursorOffset;
|
|
|
|
m_typedChar = QChar();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
if (autoInsertBrackets && data().canConvert<CompleteFunctionDeclaration>()) {
|
|
|
|
if (m_typedChar == QLatin1Char('('))
|
|
|
|
m_typedChar = QChar();
|
|
|
|
|
|
|
|
// everything from the closing parenthesis on are extra chars, to
|
|
|
|
// make sure an auto-inserted ")" gets replaced by ") const" if necessary
|
|
|
|
int closingParen = toInsert.lastIndexOf(QLatin1Char(')'));
|
|
|
|
extraChars = toInsert.mid(closingParen);
|
|
|
|
toInsert.truncate(closingParen);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
// Append an unhandled typed character, adjusting cursor offset when it had been adjusted before
|
|
|
|
if (!m_typedChar.isNull()) {
|
|
|
|
extraChars += m_typedChar;
|
|
|
|
if (cursorOffset != 0)
|
|
|
|
--cursorOffset;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Avoid inserting characters that are already there
|
2014-07-23 19:10:38 +02:00
|
|
|
const int endsPosition = editor->position(TextEditor::BaseTextEditor::EndOfLine);
|
2013-12-10 14:37:32 +01:00
|
|
|
const QString existingText = editor->textDocument()->textAt(editor->position(),
|
|
|
|
endsPosition - editor->position());
|
|
|
|
int existLength = 0;
|
|
|
|
if (!existingText.isEmpty()) {
|
|
|
|
// Calculate the exist length in front of the extra chars
|
|
|
|
existLength = toInsert.length() - (editor->position() - basePosition);
|
|
|
|
while (!existingText.startsWith(toInsert.right(existLength))) {
|
|
|
|
if (--existLength == 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int i = 0; i < extraChars.length(); ++i) {
|
|
|
|
const QChar a = extraChars.at(i);
|
|
|
|
const QChar b = editor->textDocument()->characterAt(editor->position() + i + existLength);
|
|
|
|
if (a == b)
|
|
|
|
++extraLength;
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
toInsert += extraChars;
|
|
|
|
|
|
|
|
// Insert the remainder of the name
|
|
|
|
const int length = editor->position() - basePosition + existLength + extraLength;
|
|
|
|
editor->setCursorPosition(basePosition);
|
|
|
|
editor->replace(length, toInsert);
|
|
|
|
if (cursorOffset)
|
|
|
|
editor->setCursorPosition(editor->position() + cursorOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ClangCompletionAssistInterface::objcEnabled() const
|
|
|
|
{
|
|
|
|
return m_clangWrapper->objcEnabled();
|
|
|
|
}
|
|
|
|
|
|
|
|
ClangCompletionAssistInterface::ClangCompletionAssistInterface(ClangCompleter::Ptr clangWrapper,
|
|
|
|
QTextDocument *document,
|
|
|
|
int position,
|
|
|
|
const QString &fileName,
|
|
|
|
AssistReason reason,
|
|
|
|
const QStringList &options,
|
2014-06-25 17:23:19 +02:00
|
|
|
const QList<CppTools::ProjectPart::HeaderPath> &headerPaths,
|
2013-12-10 14:37:32 +01:00
|
|
|
const PchInfo::Ptr &pchInfo)
|
|
|
|
: DefaultAssistInterface(document, position, fileName, reason)
|
|
|
|
, m_clangWrapper(clangWrapper)
|
|
|
|
, m_options(options)
|
2014-06-25 17:23:19 +02:00
|
|
|
, m_headerPaths(headerPaths)
|
2013-12-10 14:37:32 +01:00
|
|
|
, m_savedPchPointer(pchInfo)
|
|
|
|
{
|
|
|
|
Q_ASSERT(!clangWrapper.isNull());
|
|
|
|
|
|
|
|
CppModelManagerInterface *mmi = CppModelManagerInterface::instance();
|
|
|
|
Q_ASSERT(mmi);
|
|
|
|
m_unsavedFiles = Utils::createUnsavedFiles(mmi->workingCopy());
|
|
|
|
}
|
|
|
|
|
|
|
|
ClangCompletionAssistProcessor::ClangCompletionAssistProcessor()
|
|
|
|
: m_preprocessorCompletions(QStringList()
|
|
|
|
<< QLatin1String("define")
|
|
|
|
<< QLatin1String("error")
|
|
|
|
<< QLatin1String("include")
|
|
|
|
<< QLatin1String("line")
|
|
|
|
<< QLatin1String("pragma")
|
|
|
|
<< QLatin1String("pragma once")
|
|
|
|
<< QLatin1String("pragma omp atomic")
|
|
|
|
<< QLatin1String("pragma omp parallel")
|
|
|
|
<< QLatin1String("pragma omp for")
|
|
|
|
<< QLatin1String("pragma omp ordered")
|
|
|
|
<< QLatin1String("pragma omp parallel for")
|
|
|
|
<< QLatin1String("pragma omp section")
|
|
|
|
<< QLatin1String("pragma omp sections")
|
|
|
|
<< QLatin1String("pragma omp parallel sections")
|
|
|
|
<< QLatin1String("pragma omp single")
|
|
|
|
<< QLatin1String("pragma omp master")
|
|
|
|
<< QLatin1String("pragma omp critical")
|
|
|
|
<< QLatin1String("pragma omp barrier")
|
|
|
|
<< QLatin1String("pragma omp flush")
|
|
|
|
<< QLatin1String("pragma omp threadprivate")
|
|
|
|
<< QLatin1String("undef")
|
|
|
|
<< QLatin1String("if")
|
|
|
|
<< QLatin1String("ifdef")
|
|
|
|
<< QLatin1String("ifndef")
|
|
|
|
<< QLatin1String("elif")
|
|
|
|
<< QLatin1String("else")
|
|
|
|
<< QLatin1String("endif"))
|
|
|
|
, m_model(new ClangAssistProposalModel)
|
|
|
|
, m_hintProposal(0)
|
|
|
|
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
ClangCompletionAssistProcessor::~ClangCompletionAssistProcessor()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
IAssistProposal *ClangCompletionAssistProcessor::perform(const IAssistInterface *interface)
|
|
|
|
{
|
|
|
|
m_interface.reset(static_cast<const ClangCompletionAssistInterface *>(interface));
|
|
|
|
|
|
|
|
if (interface->reason() != ExplicitlyInvoked && !accepts())
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
int index = startCompletionHelper();
|
|
|
|
if (index != -1) {
|
|
|
|
if (m_hintProposal)
|
|
|
|
return m_hintProposal;
|
|
|
|
|
|
|
|
m_model->m_sortable = (m_model->m_completionOperator != T_EOF_SYMBOL);
|
|
|
|
return createContentProposal();
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ClangCompletionAssistProcessor::startCompletionHelper()
|
|
|
|
{
|
|
|
|
//### TODO: clean-up this method, some calculated values might not be used anymore.
|
|
|
|
|
|
|
|
Q_ASSERT(m_model);
|
|
|
|
|
|
|
|
const int startOfName = findStartOfName();
|
|
|
|
m_startPosition = startOfName;
|
|
|
|
m_model->m_completionOperator = T_EOF_SYMBOL;
|
|
|
|
|
|
|
|
int endOfOperator = m_startPosition;
|
|
|
|
|
|
|
|
// Skip whitespace preceding this position
|
|
|
|
while (m_interface->characterAt(endOfOperator - 1).isSpace())
|
|
|
|
--endOfOperator;
|
|
|
|
|
|
|
|
const QString fileName = m_interface->fileName();
|
|
|
|
|
|
|
|
int endOfExpression = startOfOperator(endOfOperator,
|
|
|
|
&m_model->m_completionOperator,
|
|
|
|
/*want function call =*/ true);
|
|
|
|
|
|
|
|
if (m_model->m_completionOperator == T_EOF_SYMBOL) {
|
|
|
|
endOfOperator = m_startPosition;
|
|
|
|
} else if (m_model->m_completionOperator == T_DOXY_COMMENT) {
|
|
|
|
for (int i = 1; i < T_DOXY_LAST_TAG; ++i)
|
|
|
|
addCompletionItem(QString::fromLatin1(doxygenTagSpell(i)),
|
|
|
|
m_icons.keywordIcon());
|
|
|
|
return m_startPosition;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pre-processor completion
|
|
|
|
//### TODO: check if clang can do pp completion
|
|
|
|
if (m_model->m_completionOperator == T_POUND) {
|
|
|
|
completePreprocessor();
|
|
|
|
m_startPosition = startOfName;
|
|
|
|
return m_startPosition;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Include completion
|
|
|
|
if (m_model->m_completionOperator == T_STRING_LITERAL
|
|
|
|
|| m_model->m_completionOperator == T_ANGLE_STRING_LITERAL
|
|
|
|
|| m_model->m_completionOperator == T_SLASH) {
|
|
|
|
|
|
|
|
QTextCursor c(m_interface->textDocument());
|
|
|
|
c.setPosition(endOfExpression);
|
|
|
|
if (completeInclude(c))
|
|
|
|
m_startPosition = startOfName;
|
|
|
|
return m_startPosition;
|
|
|
|
}
|
|
|
|
|
|
|
|
ExpressionUnderCursor expressionUnderCursor;
|
|
|
|
QTextCursor tc(m_interface->textDocument());
|
|
|
|
|
|
|
|
if (m_model->m_completionOperator == T_COMMA) {
|
|
|
|
tc.setPosition(endOfExpression);
|
|
|
|
const int start = expressionUnderCursor.startOfFunctionCall(tc);
|
|
|
|
if (start == -1) {
|
|
|
|
m_model->m_completionOperator = T_EOF_SYMBOL;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
endOfExpression = start;
|
|
|
|
m_startPosition = start + 1;
|
|
|
|
m_model->m_completionOperator = T_LPAREN;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString expression;
|
|
|
|
int startOfExpression = m_interface->position();
|
|
|
|
tc.setPosition(endOfExpression);
|
|
|
|
|
|
|
|
if (m_model->m_completionOperator) {
|
|
|
|
expression = expressionUnderCursor(tc);
|
|
|
|
startOfExpression = endOfExpression - expression.length();
|
|
|
|
|
|
|
|
if (m_model->m_completionOperator == T_LPAREN) {
|
|
|
|
if (expression.endsWith(QLatin1String("SIGNAL")))
|
|
|
|
m_model->m_completionOperator = T_SIGNAL;
|
|
|
|
|
|
|
|
else if (expression.endsWith(QLatin1String("SLOT")))
|
|
|
|
m_model->m_completionOperator = T_SLOT;
|
|
|
|
|
|
|
|
else if (m_interface->position() != endOfOperator) {
|
|
|
|
// We don't want a function completion when the cursor isn't at the opening brace
|
|
|
|
expression.clear();
|
|
|
|
m_model->m_completionOperator = T_EOF_SYMBOL;
|
|
|
|
m_startPosition = startOfName;
|
|
|
|
startOfExpression = m_interface->position();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (expression.isEmpty()) {
|
|
|
|
while (startOfExpression > 0 && m_interface->characterAt(startOfExpression).isSpace())
|
|
|
|
--startOfExpression;
|
|
|
|
}
|
|
|
|
|
|
|
|
int line = 0, column = 0;
|
|
|
|
// Convenience::convertPosition(m_interface->document(), startOfExpression, &line, &column);
|
|
|
|
Convenience::convertPosition(m_interface->textDocument(), endOfOperator, &line, &column);
|
|
|
|
return startCompletionInternal(fileName, line, column, endOfOperator);
|
|
|
|
}
|
|
|
|
|
|
|
|
int ClangCompletionAssistProcessor::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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*kind == T_COMMA) {
|
|
|
|
ExpressionUnderCursor expressionUnderCursor;
|
|
|
|
if (expressionUnderCursor.startOfFunctionCall(tc) == -1) {
|
|
|
|
*kind = T_EOF_SYMBOL;
|
|
|
|
start = pos;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SimpleLexer tokenize;
|
|
|
|
LanguageFeatures lf = tokenize.languageFeatures();
|
|
|
|
lf.qtMocRunEnabled = true;
|
|
|
|
lf.objCEnabled = true;
|
|
|
|
tokenize.setLanguageFeatures(lf);
|
|
|
|
tokenize.setSkipComments(false);
|
|
|
|
const QList<CPlusPlus::Token> &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 CPlusPlus::Token tk = (tokenIdx == -1) ? CPlusPlus::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 CPlusPlus::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 CPlusPlus::Token &directiveToken = tokens.at(1);
|
2013-12-13 18:41:15 +01:00
|
|
|
QString directive = tc.block().text().mid(directiveToken.bytesBegin(),
|
|
|
|
directiveToken.bytes());
|
2013-12-10 14:37:32 +01:00
|
|
|
if (directive == QLatin1String("include") ||
|
|
|
|
directive == QLatin1String("include_next") ||
|
|
|
|
directive == QLatin1String("import")) {
|
|
|
|
include = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!include) {
|
|
|
|
*kind = T_EOF_SYMBOL;
|
|
|
|
start = pos;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return start;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ClangCompletionAssistProcessor::findStartOfName(int pos) const
|
|
|
|
{
|
|
|
|
if (pos == -1)
|
|
|
|
pos = m_interface->position();
|
|
|
|
QChar chr;
|
|
|
|
|
|
|
|
// Skip to the start of a name
|
|
|
|
do {
|
|
|
|
chr = m_interface->characterAt(--pos);
|
|
|
|
} while (chr.isLetterOrNumber() || chr == QLatin1Char('_'));
|
|
|
|
|
|
|
|
return pos + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ClangCompletionAssistProcessor::accepts() const
|
|
|
|
{
|
|
|
|
const int pos = m_interface->position();
|
|
|
|
unsigned token = T_EOF_SYMBOL;
|
|
|
|
|
|
|
|
const int start = startOfOperator(pos, &token, /*want function call=*/ true);
|
|
|
|
if (start != pos) {
|
|
|
|
if (token == T_POUND) {
|
|
|
|
const int column = pos - m_interface->textDocument()->findBlock(start).position();
|
|
|
|
if (column != 1)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
// Trigger completion after three characters of a name have been typed, when not editing an existing name
|
|
|
|
QChar characterUnderCursor = m_interface->characterAt(pos);
|
|
|
|
if (!characterUnderCursor.isLetterOrNumber() && characterUnderCursor != QLatin1Char('_')) {
|
|
|
|
const int startOfName = findStartOfName(pos);
|
|
|
|
if (pos - startOfName >= 3) {
|
|
|
|
const QChar firstCharacter = m_interface->characterAt(startOfName);
|
|
|
|
if (firstCharacter.isLetter() || firstCharacter == QLatin1Char('_')) {
|
|
|
|
// Finally check that we're not inside a comment or string (code copied from startOfOperator)
|
|
|
|
QTextCursor tc(m_interface->textDocument());
|
|
|
|
tc.setPosition(pos);
|
|
|
|
|
|
|
|
SimpleLexer tokenize;
|
|
|
|
LanguageFeatures lf = tokenize.languageFeatures();
|
|
|
|
lf.qtMocRunEnabled = true;
|
|
|
|
lf.objCEnabled = true;
|
|
|
|
tokenize.setLanguageFeatures(lf);
|
|
|
|
tokenize.setSkipComments(false);
|
|
|
|
const QList<CPlusPlus::Token> &tokens = tokenize(tc.block().text(), BackwardsScanner::previousBlockState(tc.block()));
|
|
|
|
const int tokenIdx = SimpleLexer::tokenBefore(tokens, qMax(0, tc.positionInBlock() - 1));
|
|
|
|
const CPlusPlus::Token tk = (tokenIdx == -1) ? CPlusPlus::Token() : tokens.at(tokenIdx);
|
|
|
|
|
|
|
|
if (!tk.isComment() && !tk.isLiteral()) {
|
|
|
|
return true;
|
|
|
|
} else if (tk.isLiteral()
|
|
|
|
&& tokens.size() == 3
|
|
|
|
&& tokens.at(0).kind() == T_POUND
|
|
|
|
&& tokens.at(1).kind() == T_IDENTIFIER) {
|
|
|
|
const QString &line = tc.block().text();
|
|
|
|
const CPlusPlus::Token &idToken = tokens.at(1);
|
|
|
|
const QStringRef &identifier =
|
2013-12-13 18:41:15 +01:00
|
|
|
line.midRef(idToken.bytesBegin(),
|
|
|
|
idToken.bytesEnd() - idToken.bytesBegin());
|
2013-12-10 14:37:32 +01:00
|
|
|
if (identifier == QLatin1String("include")
|
|
|
|
|| identifier == QLatin1String("include_next")
|
|
|
|
|| (m_interface->objcEnabled() && identifier == QLatin1String("import"))) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
IAssistProposal *ClangCompletionAssistProcessor::createContentProposal()
|
|
|
|
{
|
|
|
|
m_model->loadContent(m_completions);
|
|
|
|
return new ClangAssistProposal(m_startPosition, m_model.take());
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Seach backwards in the document starting from pos to find the first opening
|
|
|
|
/// parenthesis. Nested parenthesis are skipped.
|
|
|
|
static int findOpenParen(QTextDocument *doc, int start)
|
|
|
|
{
|
|
|
|
unsigned parenCount = 1;
|
|
|
|
for (int pos = start; pos >= 0; --pos) {
|
|
|
|
const QChar ch = doc->characterAt(pos);
|
|
|
|
if (ch == QLatin1Char('(')) {
|
|
|
|
--parenCount;
|
|
|
|
if (parenCount == 0)
|
|
|
|
return pos;
|
|
|
|
} else if (ch == QLatin1Char(')')) {
|
|
|
|
++parenCount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static QByteArray modifyInput(QTextDocument *doc, int endOfExpression) {
|
|
|
|
int comma = endOfExpression;
|
|
|
|
while (comma > 0) {
|
|
|
|
const QChar ch = doc->characterAt(comma);
|
|
|
|
if (ch == QLatin1Char(','))
|
|
|
|
break;
|
|
|
|
if (ch == QLatin1Char(';') || ch == QLatin1Char('{') || ch == QLatin1Char('}')) {
|
|
|
|
// Safety net: we don't seem to have "connect(pointer, SIGNAL(" as
|
|
|
|
// input, so stop searching.
|
|
|
|
comma = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
--comma;
|
|
|
|
}
|
|
|
|
if (comma < 0)
|
|
|
|
return QByteArray();
|
|
|
|
const int openBrace = findOpenParen(doc, comma);
|
|
|
|
if (openBrace < 0)
|
|
|
|
return QByteArray();
|
|
|
|
|
|
|
|
QByteArray modifiedInput = doc->toPlainText().toUtf8();
|
|
|
|
const int len = endOfExpression - comma;
|
|
|
|
QByteArray replacement(len - 4, ' ');
|
|
|
|
replacement.append(")->");
|
|
|
|
modifiedInput.replace(comma, len, replacement);
|
|
|
|
modifiedInput.insert(openBrace, '(');
|
|
|
|
return modifiedInput;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ClangCompletionAssistProcessor::startCompletionInternal(const QString fileName,
|
|
|
|
unsigned line,
|
|
|
|
unsigned column,
|
|
|
|
int endOfExpression)
|
|
|
|
{
|
|
|
|
bool signalCompletion = false;
|
|
|
|
bool slotCompletion = false;
|
|
|
|
QByteArray modifiedInput;
|
|
|
|
|
|
|
|
if (m_model->m_completionOperator == T_SIGNAL) {
|
|
|
|
signalCompletion = true;
|
|
|
|
modifiedInput = modifyInput(m_interface->textDocument(), endOfExpression);
|
|
|
|
} else if (m_model->m_completionOperator == T_SLOT) {
|
|
|
|
slotCompletion = true;
|
|
|
|
modifiedInput = modifyInput(m_interface->textDocument(), endOfExpression);
|
|
|
|
} else if (m_model->m_completionOperator == T_LPAREN) {
|
|
|
|
// Find the expression that precedes the current name
|
|
|
|
int index = endOfExpression;
|
|
|
|
while (m_interface->characterAt(index - 1).isSpace())
|
|
|
|
--index;
|
|
|
|
|
|
|
|
QTextCursor tc(m_interface->textDocument());
|
|
|
|
tc.setPosition(index);
|
|
|
|
ExpressionUnderCursor euc;
|
|
|
|
index = euc.startOfFunctionCall(tc);
|
|
|
|
int nameStart = findStartOfName(index);
|
|
|
|
QTextCursor tc2(m_interface->textDocument());
|
|
|
|
tc2.setPosition(nameStart);
|
|
|
|
tc2.setPosition(index, QTextCursor::KeepAnchor);
|
|
|
|
const QString functionName = tc2.selectedText().trimmed();
|
|
|
|
int l = line, c = column;
|
|
|
|
Convenience::convertPosition(m_interface->textDocument(), nameStart, &l, &c);
|
|
|
|
|
|
|
|
if (DebugTiming)
|
|
|
|
qDebug()<<"complete constructor or function @" << line<<":"<<column << "->"<<l<<":"<<c;
|
|
|
|
|
|
|
|
const QList<CodeCompletionResult> completions = unfilteredCompletion(
|
|
|
|
m_interface.data(), fileName, l, c, QByteArray(), signalCompletion || slotCompletion);
|
|
|
|
QList<CodeCompletionResult> functionCompletions;
|
|
|
|
foreach (const CodeCompletionResult &ccr, completions) {
|
|
|
|
if (ccr.completionKind() == CodeCompletionResult::FunctionCompletionKind
|
|
|
|
|| ccr.completionKind() == CodeCompletionResult::ConstructorCompletionKind
|
|
|
|
|| ccr.completionKind() == CodeCompletionResult::DestructorCompletionKind
|
|
|
|
|| ccr.completionKind() == CodeCompletionResult::SignalCompletionKind
|
|
|
|
|| ccr.completionKind() == CodeCompletionResult::SlotCompletionKind)
|
|
|
|
if (ccr.text() == functionName)
|
|
|
|
functionCompletions.append(ccr);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!functionCompletions.isEmpty()) {
|
|
|
|
IFunctionHintProposalModel *model = new ClangFunctionHintModel(functionCompletions);
|
|
|
|
m_hintProposal = new FunctionHintProposal(m_startPosition, model);
|
|
|
|
return m_startPosition;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const QIcon snippetIcon = QIcon(QLatin1String(SNIPPET_ICON_PATH));
|
|
|
|
QList<CodeCompletionResult> completions = unfilteredCompletion(
|
|
|
|
m_interface.data(), fileName, line, column, modifiedInput, signalCompletion || slotCompletion);
|
|
|
|
QHash<QString, ClangAssistProposalItem *> items;
|
|
|
|
foreach (const CodeCompletionResult &ccr, completions) {
|
|
|
|
if (!ccr.isValid())
|
|
|
|
continue;
|
|
|
|
if (signalCompletion && ccr.completionKind() != CodeCompletionResult::SignalCompletionKind)
|
|
|
|
continue;
|
|
|
|
if (slotCompletion && ccr.completionKind() != CodeCompletionResult::SlotCompletionKind)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
const QString txt(ccr.text());
|
|
|
|
ClangAssistProposalItem *item = items.value(txt, 0);
|
|
|
|
if (item) {
|
|
|
|
item->addOverload(ccr);
|
|
|
|
} else {
|
|
|
|
item = new ClangAssistProposalItem;
|
|
|
|
items.insert(txt, item);
|
|
|
|
item->setText(txt);
|
|
|
|
item->setDetail(ccr.hint());
|
|
|
|
item->setOrder(ccr.priority());
|
|
|
|
|
|
|
|
const QString snippet = ccr.snippet();
|
|
|
|
if (!snippet.isEmpty())
|
|
|
|
item->setData(snippet);
|
|
|
|
else
|
|
|
|
item->setData(qVariantFromValue(ccr));
|
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: show the effective accessebility instead of availability
|
|
|
|
switch (ccr.completionKind()) {
|
|
|
|
case CodeCompletionResult::ClassCompletionKind: item->setIcon(m_icons.iconForType(Icons::ClassIconType)); break;
|
|
|
|
case CodeCompletionResult::EnumCompletionKind: item->setIcon(m_icons.iconForType(Icons::EnumIconType)); break;
|
|
|
|
case CodeCompletionResult::EnumeratorCompletionKind: item->setIcon(m_icons.iconForType(Icons::EnumeratorIconType)); break;
|
|
|
|
|
|
|
|
case CodeCompletionResult::ConstructorCompletionKind: // fall through
|
|
|
|
case CodeCompletionResult::DestructorCompletionKind: // fall through
|
|
|
|
case CodeCompletionResult::FunctionCompletionKind:
|
|
|
|
case CodeCompletionResult::ObjCMessageCompletionKind:
|
|
|
|
switch (ccr.availability()) {
|
|
|
|
case CodeCompletionResult::Available:
|
|
|
|
case CodeCompletionResult::Deprecated:
|
|
|
|
item->setIcon(m_icons.iconForType(Icons::FuncPublicIconType));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
item->setIcon(m_icons.iconForType(Icons::FuncPrivateIconType));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CodeCompletionResult::SignalCompletionKind:
|
|
|
|
item->setIcon(m_icons.iconForType(Icons::SignalIconType));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CodeCompletionResult::SlotCompletionKind:
|
|
|
|
switch (ccr.availability()) {
|
|
|
|
case CodeCompletionResult::Available:
|
|
|
|
case CodeCompletionResult::Deprecated:
|
|
|
|
item->setIcon(m_icons.iconForType(Icons::SlotPublicIconType));
|
|
|
|
break;
|
|
|
|
case CodeCompletionResult::NotAccessible:
|
|
|
|
case CodeCompletionResult::NotAvailable:
|
|
|
|
item->setIcon(m_icons.iconForType(Icons::SlotPrivateIconType));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CodeCompletionResult::NamespaceCompletionKind: item->setIcon(m_icons.iconForType(Icons::NamespaceIconType)); break;
|
|
|
|
case CodeCompletionResult::PreProcessorCompletionKind: item->setIcon(m_icons.iconForType(Icons::MacroIconType)); break;
|
|
|
|
case CodeCompletionResult::VariableCompletionKind:
|
|
|
|
switch (ccr.availability()) {
|
|
|
|
case CodeCompletionResult::Available:
|
|
|
|
case CodeCompletionResult::Deprecated:
|
|
|
|
item->setIcon(m_icons.iconForType(Icons::VarPublicIconType));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
item->setIcon(m_icons.iconForType(Icons::VarPrivateIconType));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CodeCompletionResult::KeywordCompletionKind:
|
|
|
|
item->setIcon(m_icons.iconForType(Icons::KeywordIconType));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CodeCompletionResult::ClangSnippetKind:
|
|
|
|
item->setIcon(snippetIcon);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (ClangAssistProposalItem *item, items.values())
|
|
|
|
m_completions.append(item);
|
|
|
|
|
|
|
|
return m_startPosition;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Creates completion proposals for #include and given cursor
|
|
|
|
* @param cursor - cursor placed after opening bracked or quote
|
|
|
|
* @return false if completions list is empty
|
|
|
|
*/
|
|
|
|
bool ClangCompletionAssistProcessor::completeInclude(const QTextCursor &cursor)
|
|
|
|
{
|
|
|
|
QString directoryPrefix;
|
|
|
|
if (m_model->m_completionOperator == T_SLASH) {
|
|
|
|
QTextCursor c = cursor;
|
|
|
|
c.movePosition(QTextCursor::StartOfLine, QTextCursor::KeepAnchor);
|
|
|
|
QString sel = c.selectedText();
|
|
|
|
int startCharPos = sel.indexOf(QLatin1Char('"'));
|
|
|
|
if (startCharPos == -1) {
|
|
|
|
startCharPos = sel.indexOf(QLatin1Char('<'));
|
|
|
|
m_model->m_completionOperator = T_ANGLE_STRING_LITERAL;
|
|
|
|
} else {
|
|
|
|
m_model->m_completionOperator = T_STRING_LITERAL;
|
|
|
|
}
|
|
|
|
if (startCharPos != -1)
|
|
|
|
directoryPrefix = sel.mid(startCharPos + 1, sel.length() - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make completion for all relevant includes
|
2014-06-25 17:23:19 +02:00
|
|
|
ProjectPart::HeaderPaths headerPaths = m_interface->headerPaths();
|
|
|
|
const ProjectPart::HeaderPath currentFilePath(QFileInfo(m_interface->fileName()).path(),
|
|
|
|
ProjectPart::HeaderPath::IncludePath);
|
|
|
|
if (!headerPaths.contains(currentFilePath))
|
|
|
|
headerPaths.append(currentFilePath);
|
2013-12-10 14:37:32 +01:00
|
|
|
|
|
|
|
const Core::MimeType mimeType = Core::MimeDatabase::findByType(QLatin1String("text/x-c++hdr"));
|
|
|
|
const QStringList suffixes = mimeType.suffixes();
|
|
|
|
|
2014-06-25 17:23:19 +02:00
|
|
|
foreach (const ProjectPart::HeaderPath &headerPath, headerPaths) {
|
|
|
|
QString realPath = headerPath.path;
|
2013-12-10 14:37:32 +01:00
|
|
|
if (!directoryPrefix.isEmpty()) {
|
|
|
|
realPath += QLatin1Char('/');
|
|
|
|
realPath += directoryPrefix;
|
2014-06-25 17:23:19 +02:00
|
|
|
if (headerPath.isFrameworkPath())
|
|
|
|
realPath += QLatin1String(".framework/Headers");
|
2013-12-10 14:37:32 +01:00
|
|
|
}
|
|
|
|
completeIncludePath(realPath, suffixes);
|
|
|
|
}
|
|
|
|
|
|
|
|
return !m_completions.isEmpty();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Adds #include completion proposals using given include path
|
|
|
|
* @param realPath - one of directories where compiler searches includes
|
|
|
|
* @param suffixes - file suffixes for C/C++ header files
|
|
|
|
*/
|
|
|
|
void ClangCompletionAssistProcessor::completeIncludePath(const QString &realPath,
|
|
|
|
const QStringList &suffixes)
|
|
|
|
{
|
|
|
|
QDirIterator i(realPath, QDir::Files | QDir::Dirs | QDir::NoDotAndDotDot);
|
2014-03-03 15:53:34 +01:00
|
|
|
//: Parent folder for proposed #include completion
|
|
|
|
const QString hint = tr("Location: %1").arg(QDir::toNativeSeparators(QDir::cleanPath(realPath)));
|
2013-12-10 14:37:32 +01:00
|
|
|
while (i.hasNext()) {
|
|
|
|
const QString fileName = i.next();
|
|
|
|
const QFileInfo fileInfo = i.fileInfo();
|
|
|
|
const QString suffix = fileInfo.suffix();
|
|
|
|
if (suffix.isEmpty() || suffixes.contains(suffix)) {
|
|
|
|
QString text = fileName.mid(realPath.length() + 1);
|
|
|
|
if (fileInfo.isDir())
|
|
|
|
text += QLatin1Char('/');
|
|
|
|
|
|
|
|
ClangAssistProposalItem *item = new ClangAssistProposalItem;
|
|
|
|
item->setText(text);
|
|
|
|
item->setDetail(hint);
|
|
|
|
item->setIcon(m_icons.keywordIcon());
|
|
|
|
item->keepCompletionOperator(m_model->m_completionOperator);
|
|
|
|
m_completions.append(item);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ClangCompletionAssistProcessor::completePreprocessor()
|
|
|
|
{
|
|
|
|
foreach (const QString &preprocessorCompletion, m_preprocessorCompletions)
|
|
|
|
addCompletionItem(preprocessorCompletion,
|
|
|
|
m_icons.iconForType(Icons::MacroIconType));
|
|
|
|
|
|
|
|
if (m_interface->objcEnabled())
|
|
|
|
addCompletionItem(QLatin1String("import"),
|
|
|
|
m_icons.iconForType(Icons::MacroIconType));
|
|
|
|
}
|
|
|
|
|
|
|
|
void ClangCompletionAssistProcessor::addCompletionItem(const QString &text,
|
|
|
|
const QIcon &icon,
|
|
|
|
int order,
|
|
|
|
const QVariant &data)
|
|
|
|
{
|
|
|
|
ClangAssistProposalItem *item = new ClangAssistProposalItem;
|
|
|
|
item->setText(text);
|
|
|
|
item->setIcon(icon);
|
|
|
|
item->setOrder(order);
|
|
|
|
item->setData(data);
|
|
|
|
item->keepCompletionOperator(m_model->m_completionOperator);
|
|
|
|
m_completions.append(item);
|
|
|
|
}
|