Clang: Remember selected function signature hint

...when typing more arguments:

    struct Foo {};
    void f(int, int);
    void f(Foo, Foo);
    void f(char, char);

    void c()
    {
        f( // 1. Trigger completion with Ctrl+Space
           // 2. Chose item "f(Foo, Foo)"
           // 3. Type: Foo(),
           // OK, signature hint "f(Foo, Foo)" is displayed again
    }

FunctionHintProposalWidget and IFunctionHintProposalModel are
instantiated for each calculation, so remember the selected hint in the
CodeAssist. Keep the latest 20 entries.

Task-number: QTCREATORBUG-11688
Change-Id: I579fc6d8a35dd8fa398e4b3170ddc05a85252d1a
Reviewed-by: Tim Jenssen <tim.jenssen@qt.io>
This commit is contained in:
Nikolai Kosjar
2017-08-18 12:21:45 +02:00
parent 0ff2c036ac
commit b4e2ab36a7
10 changed files with 151 additions and 1 deletions

View File

@@ -61,6 +61,15 @@ QString ClangFunctionHintModel::text(int index) const
return signatureWithEmphasizedCurrentParameter; return signatureWithEmphasizedCurrentParameter;
} }
QString ClangFunctionHintModel::id(int index) const
{
QString chunks;
for (const ClangBackEnd::CodeCompletionChunk &chunk : m_functionSymbols.at(index).chunks())
chunks += chunk.text();
return chunks;
}
int ClangFunctionHintModel::activeArgument(const QString &prefix) const int ClangFunctionHintModel::activeArgument(const QString &prefix) const
{ {
int activeArgumentNumber = 0; int activeArgumentNumber = 0;

View File

@@ -40,6 +40,7 @@ public:
void reset() override; void reset() override;
int size() const override; int size() const override;
QString text(int index) const override; QString text(int index) const override;
QString id(int index) const override;
int activeArgument(const QString &prefix) const override; int activeArgument(const QString &prefix) const override;
private: private:

View File

@@ -74,6 +74,9 @@ public:
bool hasContext() const; bool hasContext() const;
void destroyContext(); void destroyContext();
QVariant userData() const;
void setUserData(const QVariant &data);
CompletionAssistProvider *identifyActivationSequence(); CompletionAssistProvider *identifyActivationSequence();
void stopAutomaticProposalTimer(); void stopAutomaticProposalTimer();
@@ -105,6 +108,7 @@ private:
CompletionSettings m_settings; CompletionSettings m_settings;
int m_abortedBasePosition; int m_abortedBasePosition;
static const QChar m_null; static const QChar m_null;
QVariant m_userData;
}; };
// -------------------- // --------------------
@@ -340,6 +344,7 @@ void CodeAssistantPrivate::displayProposal(IAssistProposal *newProposal, AssistR
m_proposalWidget->setAssistant(q); m_proposalWidget->setAssistant(q);
m_proposalWidget->setReason(reason); m_proposalWidget->setReason(reason);
m_proposalWidget->setKind(m_assistKind); m_proposalWidget->setKind(m_assistKind);
m_proposalWidget->setBasePosition(basePosition);
m_proposalWidget->setUnderlyingWidget(m_editorWidget); m_proposalWidget->setUnderlyingWidget(m_editorWidget);
m_proposalWidget->setModel(proposalCandidateModel.take()); m_proposalWidget->setModel(proposalCandidateModel.take());
m_proposalWidget->setDisplayRect(m_editorWidget->cursorRect(basePosition)); m_proposalWidget->setDisplayRect(m_editorWidget->cursorRect(basePosition));
@@ -465,6 +470,16 @@ void CodeAssistantPrivate::destroyContext()
} }
} }
QVariant CodeAssistantPrivate::userData() const
{
return m_userData;
}
void CodeAssistantPrivate::setUserData(const QVariant &data)
{
m_userData = data;
}
void CodeAssistantPrivate::startAutomaticProposalTimer() void CodeAssistantPrivate::startAutomaticProposalTimer()
{ {
if (m_settings.m_completionTrigger == AutomaticCompletion) if (m_settings.m_completionTrigger == AutomaticCompletion)
@@ -572,6 +587,16 @@ void CodeAssistant::destroyContext()
d->destroyContext(); d->destroyContext();
} }
QVariant CodeAssistant::userData() const
{
return d->userData();
}
void CodeAssistant::setUserData(const QVariant &data)
{
d->setUserData(data);
}
void CodeAssistant::invoke(AssistKind kind, IAssistProvider *provider) void CodeAssistant::invoke(AssistKind kind, IAssistProvider *provider)
{ {
d->invoke(kind, provider); d->invoke(kind, provider);

View File

@@ -53,6 +53,9 @@ public:
bool hasContext() const; bool hasContext() const;
void destroyContext(); void destroyContext();
QVariant userData() const;
void setUserData(const QVariant &data);
void invoke(AssistKind assistKind, IAssistProvider *provider = 0); void invoke(AssistKind assistKind, IAssistProvider *provider = 0);
signals: signals:

View File

@@ -27,6 +27,7 @@
#include "ifunctionhintproposalmodel.h" #include "ifunctionhintproposalmodel.h"
#include "codeassistant.h" #include "codeassistant.h"
#include <utils/algorithm.h>
#include <utils/faketooltip.h> #include <utils/faketooltip.h>
#include <utils/hostosinfo.h> #include <utils/hostosinfo.h>
#include <utils/qtcassert.h> #include <utils/qtcassert.h>
@@ -42,6 +43,56 @@
namespace TextEditor { namespace TextEditor {
static const int maxSelectedFunctionHints = 20;
class SelectedFunctionHints
{
public:
void insert(int basePosition, const QString &hintId)
{
if (basePosition < 0 || hintId.isEmpty())
return;
const int index = indexOf(basePosition);
// Add new item
if (index == -1) {
if (m_items.size() + 1 > maxSelectedFunctionHints)
m_items.removeLast();
m_items.prepend(FunctionHintItem(basePosition, hintId));
return;
}
// Update existing item
m_items[index].hintId = hintId;
}
QString hintId(int basePosition) const
{
const int index = indexOf(basePosition);
return index == -1 ? QString() : m_items.at(index).hintId;
}
private:
int indexOf(int basePosition) const
{
return Utils::indexOf(m_items, [&](const FunctionHintItem &item) {
return item.basePosition == basePosition;
});
}
private:
struct FunctionHintItem {
FunctionHintItem(int basePosition, const QString &hintId)
: basePosition(basePosition), hintId(hintId) {}
int basePosition = -1;
QString hintId;
};
QList<FunctionHintItem> m_items;
};
// ------------------------- // -------------------------
// HintProposalWidgetPrivate // HintProposalWidgetPrivate
// ------------------------- // -------------------------
@@ -158,7 +209,7 @@ void FunctionHintProposalWidget::showProposal(const QString &prefix)
QTC_ASSERT(d->m_totalHints != 0, abort(); return; ); QTC_ASSERT(d->m_totalHints != 0, abort(); return; );
d->m_pager->setVisible(d->m_totalHints > 1); d->m_pager->setVisible(d->m_totalHints > 1);
d->m_currentHint = 0; d->m_currentHint = loadSelectedHint();
if (!updateAndCheck(prefix)) if (!updateAndCheck(prefix))
return; return;
@@ -184,6 +235,32 @@ void FunctionHintProposalWidget::abort()
deleteLater(); deleteLater();
} }
static SelectedFunctionHints selectedFunctionHints(CodeAssistant &codeAssistant)
{
const QVariant variant = codeAssistant.userData();
return variant.value<SelectedFunctionHints>();
}
int FunctionHintProposalWidget::loadSelectedHint() const
{
const QString hintId = selectedFunctionHints(*d->m_assistant).hintId(basePosition());
for (int i = 0; i < d->m_model->size(); ++i) {
if (d->m_model->id(i) == hintId)
return i;
}
return 0;
}
void FunctionHintProposalWidget::storeSelectedHint()
{
SelectedFunctionHints table = selectedFunctionHints(*d->m_assistant);
table.insert(basePosition(), d->m_model->id(d->m_currentHint));
d->m_assistant->setUserData(QVariant::fromValue(table));
}
bool FunctionHintProposalWidget::eventFilter(QObject *obj, QEvent *e) bool FunctionHintProposalWidget::eventFilter(QObject *obj, QEvent *e)
{ {
switch (e->type()) { switch (e->type()) {
@@ -257,6 +334,8 @@ bool FunctionHintProposalWidget::eventFilter(QObject *obj, QEvent *e)
void FunctionHintProposalWidget::nextPage() void FunctionHintProposalWidget::nextPage()
{ {
d->m_currentHint = (d->m_currentHint + 1) % d->m_totalHints; d->m_currentHint = (d->m_currentHint + 1) % d->m_totalHints;
storeSelectedHint();
updateContent(); updateContent();
} }
@@ -266,6 +345,8 @@ void FunctionHintProposalWidget::previousPage()
d->m_currentHint = d->m_totalHints - 1; d->m_currentHint = d->m_totalHints - 1;
else else
--d->m_currentHint; --d->m_currentHint;
storeSelectedHint();
updateContent(); updateContent();
} }
@@ -322,3 +403,5 @@ void FunctionHintProposalWidget::updatePosition()
} }
} // TextEditor } // TextEditor
Q_DECLARE_METATYPE(TextEditor::SelectedFunctionHints)

View File

@@ -63,6 +63,9 @@ private:
void updatePosition(); void updatePosition();
void abort(); void abort();
int loadSelectedHint() const;
void storeSelectedHint();
private: private:
FunctionHintProposalWidgetPrivate *d; FunctionHintProposalWidgetPrivate *d;
}; };

View File

@@ -55,6 +55,16 @@ IAssistProposalWidget::IAssistProposalWidget()
IAssistProposalWidget::~IAssistProposalWidget() IAssistProposalWidget::~IAssistProposalWidget()
{} {}
int IAssistProposalWidget::basePosition() const
{
return m_basePosition;
}
void IAssistProposalWidget::setBasePosition(int basePosition)
{
m_basePosition = basePosition;
}
/*! /*!
\fn void TextEditor::IAssistProposalWidget::setAssistant(CodeAssistant *assistant) \fn void TextEditor::IAssistProposalWidget::setAssistant(CodeAssistant *assistant)

View File

@@ -57,10 +57,16 @@ public:
virtual void updateProposal(const QString &prefix) = 0; virtual void updateProposal(const QString &prefix) = 0;
virtual void closeProposal() = 0; virtual void closeProposal() = 0;
int basePosition() const;
void setBasePosition(int basePosition);
signals: signals:
void prefixExpanded(const QString &newPrefix); void prefixExpanded(const QString &newPrefix);
void proposalItemActivated(AssistProposalItemInterface *proposalItem); void proposalItemActivated(AssistProposalItemInterface *proposalItem);
void explicitlyAborted(); void explicitlyAborted();
protected:
int m_basePosition = -1;
}; };
} // TextEditor } // TextEditor

View File

@@ -25,6 +25,8 @@
#include "ifunctionhintproposalmodel.h" #include "ifunctionhintproposalmodel.h"
#include <QString>
using namespace TextEditor; using namespace TextEditor;
IFunctionHintProposalModel::IFunctionHintProposalModel() IFunctionHintProposalModel::IFunctionHintProposalModel()
@@ -32,3 +34,8 @@ IFunctionHintProposalModel::IFunctionHintProposalModel()
IFunctionHintProposalModel::~IFunctionHintProposalModel() IFunctionHintProposalModel::~IFunctionHintProposalModel()
{} {}
QString IFunctionHintProposalModel::id(int /*index*/) const
{
return QString();
}

View File

@@ -29,6 +29,8 @@
#include <texteditor/texteditor_global.h> #include <texteditor/texteditor_global.h>
QT_FORWARD_DECLARE_CLASS(QString);
namespace TextEditor { namespace TextEditor {
class TEXTEDITOR_EXPORT IFunctionHintProposalModel : public IAssistProposalModel class TEXTEDITOR_EXPORT IFunctionHintProposalModel : public IAssistProposalModel
@@ -38,6 +40,7 @@ public:
~IFunctionHintProposalModel(); ~IFunctionHintProposalModel();
virtual int activeArgument(const QString &prefix) const = 0; virtual int activeArgument(const QString &prefix) const = 0;
virtual QString id(int index) const;
}; };
} // TextEditor } // TextEditor