forked from qt-creator/qt-creator
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:
@@ -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;
|
||||||
|
@@ -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:
|
||||||
|
@@ -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);
|
||||||
|
@@ -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:
|
||||||
|
@@ -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)
|
||||||
|
@@ -63,6 +63,9 @@ private:
|
|||||||
void updatePosition();
|
void updatePosition();
|
||||||
void abort();
|
void abort();
|
||||||
|
|
||||||
|
int loadSelectedHint() const;
|
||||||
|
void storeSelectedHint();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
FunctionHintProposalWidgetPrivate *d;
|
FunctionHintProposalWidgetPrivate *d;
|
||||||
};
|
};
|
||||||
|
@@ -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)
|
||||||
|
|
||||||
|
@@ -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
|
||||||
|
@@ -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();
|
||||||
|
}
|
||||||
|
@@ -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
|
||||||
|
Reference in New Issue
Block a user