2021-08-23 21:50:53 +08:00
|
|
|
#include "codecompletionview.h"
|
2021-08-24 15:05:10 +08:00
|
|
|
#include "../utils.h"
|
2021-08-23 21:50:53 +08:00
|
|
|
|
2021-08-24 09:59:44 +08:00
|
|
|
#include <QKeyEvent>
|
2021-08-24 15:05:10 +08:00
|
|
|
#include <QVBoxLayout>
|
2021-08-27 16:38:55 +08:00
|
|
|
#include <QDebug>
|
2021-08-24 09:59:44 +08:00
|
|
|
|
2021-08-23 21:50:53 +08:00
|
|
|
CodeCompletionView::CodeCompletionView(QWidget *parent) :
|
2021-08-24 09:59:44 +08:00
|
|
|
QWidget(parent)
|
2021-08-23 21:50:53 +08:00
|
|
|
{
|
2021-08-24 09:59:44 +08:00
|
|
|
setWindowFlags(Qt::Popup);
|
|
|
|
mListView = new CodeCompletionListView(this);
|
2021-08-25 08:48:33 +08:00
|
|
|
mModel=new CodeCompletionListModel(&mCompletionStatementList);
|
|
|
|
mListView->setModel(mModel);
|
2021-08-24 09:59:44 +08:00
|
|
|
setLayout(new QVBoxLayout());
|
|
|
|
layout()->addWidget(mListView);
|
|
|
|
layout()->setMargin(0);
|
2021-08-25 00:20:07 +08:00
|
|
|
|
|
|
|
mShowKeywords=true;
|
|
|
|
mUseCppKeyword=true;
|
2021-08-27 16:38:55 +08:00
|
|
|
mRecordUsage = false;
|
|
|
|
mSortByScope = true;
|
2021-08-25 00:20:07 +08:00
|
|
|
|
|
|
|
mOnlyGlobals = false;
|
|
|
|
mShowCount = 1000;
|
|
|
|
mShowCodeIns = true;
|
|
|
|
|
|
|
|
mIgnoreCase = false;
|
2021-08-25 08:48:33 +08:00
|
|
|
|
2021-08-23 21:50:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
CodeCompletionView::~CodeCompletionView()
|
|
|
|
{
|
2021-08-25 08:48:33 +08:00
|
|
|
delete mListView;
|
|
|
|
delete mModel;
|
2021-08-24 09:59:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void CodeCompletionView::setKeypressedCallback(const KeyPressedCallback &newKeypressedCallback)
|
|
|
|
{
|
|
|
|
mListView->setKeypressedCallback(newKeypressedCallback);
|
|
|
|
}
|
|
|
|
|
2021-08-25 00:20:07 +08:00
|
|
|
void CodeCompletionView::prepareSearch(const QString &phrase, const QString &filename, int line)
|
|
|
|
{
|
|
|
|
QMutexLocker locker(&mMutex);
|
2021-08-25 08:48:33 +08:00
|
|
|
if (!isEnabled())
|
2021-08-25 00:20:07 +08:00
|
|
|
return;
|
|
|
|
mPhrase = phrase;
|
|
|
|
//Screen.Cursor := crHourglass;
|
|
|
|
QCursor oldCursor = cursor();
|
|
|
|
setCursor(Qt::CursorShape::WaitCursor);
|
|
|
|
|
|
|
|
mIncludedFiles = mParser->getFileIncludes(filename);
|
|
|
|
getCompletionFor(filename,phrase,line);
|
|
|
|
|
|
|
|
//todo: notify model
|
|
|
|
//CodeComplForm.lbCompletion.Font.Size := FontSize;
|
|
|
|
//CodeComplForm.lbCompletion.ItemHeight := CodeComplForm.lbCompletion.Canvas.TextHeight('F')+6;
|
|
|
|
// Round(2 * FontSize);
|
|
|
|
//CodeComplForm.Update;
|
|
|
|
setCursor(oldCursor);
|
|
|
|
}
|
|
|
|
|
2021-08-25 08:48:33 +08:00
|
|
|
bool CodeCompletionView::search(const QString &phrase, bool autoHideOnSingleResult)
|
2021-08-25 00:20:07 +08:00
|
|
|
{
|
|
|
|
QMutexLocker locker(&mMutex);
|
|
|
|
|
|
|
|
mPhrase = phrase;
|
|
|
|
|
|
|
|
if (phrase.isEmpty()) {
|
|
|
|
hide();
|
|
|
|
return false;
|
|
|
|
}
|
2021-08-25 08:48:33 +08:00
|
|
|
if (!isEnabled()) {
|
2021-08-25 00:20:07 +08:00
|
|
|
hide();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QCursor oldCursor = cursor();
|
|
|
|
setCursor(Qt::CursorShape::WaitCursor);
|
|
|
|
|
|
|
|
// Sort here by member
|
|
|
|
int i = mParser->findLastOperator(phrase);
|
|
|
|
while ((i>=0) && (i<phrase.length()) && (
|
|
|
|
phrase[i] == '.'
|
|
|
|
|| phrase[i] == ':'
|
|
|
|
|| phrase[i] == '-'
|
|
|
|
|| phrase[i] == '>'))
|
|
|
|
i++;
|
|
|
|
|
|
|
|
QString symbol = phrase.mid(i);
|
|
|
|
// filter fFullCompletionStatementList to fCompletionStatementList
|
|
|
|
filterList(symbol);
|
2021-08-25 08:48:33 +08:00
|
|
|
mModel->notifyUpdated();
|
|
|
|
setCursor(oldCursor);
|
2021-08-25 00:20:07 +08:00
|
|
|
|
|
|
|
if (!mCompletionStatementList.isEmpty()) {
|
|
|
|
// if only one suggestion, and is exactly the symbol to search, hide the frame (the search is over)
|
|
|
|
// if only one suggestion and auto hide , don't show the frame
|
|
|
|
if(mCompletionStatementList.count() == 1)
|
|
|
|
if (autoHideOnSingleResult
|
|
|
|
|| (symbol == mCompletionStatementList.front()->command)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
hide();
|
|
|
|
}
|
2021-08-25 08:48:33 +08:00
|
|
|
return false;
|
2021-08-25 00:20:07 +08:00
|
|
|
}
|
|
|
|
|
2021-08-26 17:48:23 +08:00
|
|
|
PStatement CodeCompletionView::selectedStatement()
|
|
|
|
{
|
|
|
|
if (isEnabled()) {
|
|
|
|
int index = mListView->currentIndex().row();
|
|
|
|
if (mListView->currentIndex().isValid()
|
|
|
|
&& (index<mCompletionStatementList.count()) ) {
|
|
|
|
return mCompletionStatementList[index];
|
|
|
|
} else {
|
|
|
|
if (!mCompletionStatementList.isEmpty())
|
|
|
|
return mCompletionStatementList.front();
|
|
|
|
else
|
|
|
|
return PStatement();
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
return PStatement();
|
|
|
|
}
|
|
|
|
|
2021-08-24 15:05:10 +08:00
|
|
|
void CodeCompletionView::addChildren(PStatement scopeStatement, const QString &fileName, int line)
|
|
|
|
{
|
|
|
|
if (scopeStatement && !isIncluded(scopeStatement->fileName)
|
|
|
|
&& !isIncluded(scopeStatement->definitionFileName))
|
|
|
|
return;
|
|
|
|
const StatementMap& children = mParser->statementList().childrenStatements(scopeStatement);
|
|
|
|
if (children.isEmpty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!scopeStatement) { //Global scope
|
|
|
|
for (PStatement childStatement: children) {
|
|
|
|
if (childStatement->fileName.isEmpty()) {
|
|
|
|
// hard defines
|
|
|
|
addStatement(childStatement,fileName,-1);
|
|
|
|
} else if (!( childStatement->kind == StatementKind::skConstructor
|
|
|
|
|| childStatement->kind == StatementKind::skDestructor
|
|
|
|
|| childStatement->kind == StatementKind::skBlock)
|
|
|
|
&& (!mAddedStatements.contains(childStatement->command))
|
|
|
|
&& (
|
|
|
|
isIncluded(childStatement->fileName)
|
|
|
|
|| isIncluded(childStatement->definitionFileName)
|
|
|
|
)
|
|
|
|
) {
|
|
|
|
//we must check if the statement is included by the file
|
|
|
|
addStatement(childStatement,fileName,line);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (PStatement childStatement: children) {
|
|
|
|
if (!( childStatement->kind == StatementKind::skConstructor
|
|
|
|
|| childStatement->kind == StatementKind::skDestructor
|
|
|
|
|| childStatement->kind == StatementKind::skBlock)
|
|
|
|
&& (!mAddedStatements.contains(childStatement->command)))
|
|
|
|
addStatement(childStatement,fileName,line);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeCompletionView::addStatement(PStatement statement, const QString &fileName, int line)
|
|
|
|
{
|
2021-08-25 00:20:07 +08:00
|
|
|
if (mAddedStatements.contains(statement->command))
|
|
|
|
return;
|
2021-08-24 15:05:10 +08:00
|
|
|
if ((line!=-1)
|
|
|
|
&& (line < statement->line)
|
|
|
|
&& (fileName == statement->fileName))
|
|
|
|
return;
|
|
|
|
mAddedStatements.insert(statement->command);
|
|
|
|
mFullCompletionStatementList.append(statement);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool defaultComparator(PStatement statement1,PStatement statement2) {
|
|
|
|
// Show user template first
|
|
|
|
if (statement1->kind == StatementKind::skUserCodeIn) {
|
|
|
|
if (statement2->kind != StatementKind::skUserCodeIn)
|
|
|
|
return true;
|
|
|
|
else
|
|
|
|
return statement1->command < statement2->command;
|
|
|
|
} else if (statement2->kind == StatementKind::skUserCodeIn) {
|
|
|
|
return false;
|
|
|
|
// show keywords first
|
|
|
|
} else if ((statement1->kind == StatementKind::skKeyword)
|
|
|
|
&& (statement2->kind != StatementKind::skKeyword)) {
|
|
|
|
return true;
|
|
|
|
} else if ((statement1->kind != StatementKind::skKeyword)
|
|
|
|
&& (statement2->kind == StatementKind::skKeyword)) {
|
|
|
|
return false;
|
|
|
|
} else
|
|
|
|
return statement1->command < statement2->command;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool sortByScopeComparator(PStatement statement1,PStatement statement2){
|
|
|
|
// Show user template first
|
|
|
|
if (statement1->kind == StatementKind::skUserCodeIn) {
|
|
|
|
if (statement2->kind != StatementKind::skUserCodeIn)
|
|
|
|
return true;
|
|
|
|
else
|
|
|
|
return statement1->command < statement2->command;
|
|
|
|
} else if (statement2->kind == StatementKind::skUserCodeIn) {
|
|
|
|
return false;
|
|
|
|
// show keywords first
|
|
|
|
} else if (statement1->kind == StatementKind::skKeyword) {
|
|
|
|
if (statement2->kind != StatementKind::skKeyword)
|
|
|
|
return true;
|
|
|
|
else
|
|
|
|
return statement1->command < statement2->command;
|
|
|
|
} else if (statement2->kind == StatementKind::skKeyword) {
|
|
|
|
return false;
|
|
|
|
// Show stuff from local headers first
|
2021-08-27 16:38:55 +08:00
|
|
|
} else if (statement1->inSystemHeader && (!statement2->inSystemHeader)) {
|
2021-08-24 15:05:10 +08:00
|
|
|
return true;
|
2021-08-27 16:38:55 +08:00
|
|
|
} else if ((!statement1->inSystemHeader) && statement2->inSystemHeader) {
|
2021-08-24 15:05:10 +08:00
|
|
|
return false;
|
|
|
|
// Show local statements first
|
|
|
|
} else if (statement1->scope != StatementScope::ssGlobal
|
2021-08-27 16:38:55 +08:00
|
|
|
&& statement2->scope == StatementScope::ssGlobal ) {
|
2021-08-24 15:05:10 +08:00
|
|
|
return true;
|
|
|
|
} else if (statement1->scope == StatementScope::ssGlobal
|
2021-08-27 16:38:55 +08:00
|
|
|
&& statement2->scope != StatementScope::ssGlobal ) {
|
2021-08-24 15:05:10 +08:00
|
|
|
return false;
|
|
|
|
// otherwise, sort by name
|
|
|
|
} else
|
|
|
|
return statement1->command < statement2->command;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool sortWithUsageComparator(PStatement statement1,PStatement statement2) {
|
|
|
|
// Show user template first
|
|
|
|
if (statement1->kind == StatementKind::skUserCodeIn) {
|
|
|
|
if (statement2->kind != StatementKind::skUserCodeIn)
|
|
|
|
return true;
|
|
|
|
else
|
|
|
|
return statement1->command < statement2->command;
|
|
|
|
} else if (statement2->kind == StatementKind::skUserCodeIn) {
|
|
|
|
return false;
|
|
|
|
//show most freq first
|
|
|
|
} else if (statement1->freqTop > statement2->freqTop) {
|
|
|
|
return true;
|
|
|
|
} else if (statement1->freqTop < statement2->freqTop) {
|
|
|
|
return false;
|
|
|
|
// show keywords first
|
|
|
|
} else if ((statement1->kind == StatementKind::skKeyword)
|
|
|
|
&& (statement2->kind != StatementKind::skKeyword)) {
|
|
|
|
return true;
|
|
|
|
} else if ((statement1->kind != StatementKind::skKeyword)
|
|
|
|
&& (statement2->kind == StatementKind::skKeyword)) {
|
|
|
|
return false;
|
|
|
|
} else
|
|
|
|
return statement1->command < statement2->command;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool sortByScopeWithUsageComparator(PStatement statement1,PStatement statement2){
|
|
|
|
// Show user template first
|
|
|
|
if (statement1->kind == StatementKind::skUserCodeIn) {
|
|
|
|
if (statement2->kind != StatementKind::skUserCodeIn)
|
|
|
|
return true;
|
|
|
|
else
|
|
|
|
return statement1->command < statement2->command;
|
|
|
|
} else if (statement2->kind == StatementKind::skUserCodeIn) {
|
|
|
|
return false;
|
|
|
|
//show most freq first
|
|
|
|
} else if (statement1->freqTop > statement2->freqTop) {
|
|
|
|
return true;
|
|
|
|
} else if (statement1->freqTop < statement2->freqTop) {
|
|
|
|
return false;
|
|
|
|
// show keywords first
|
|
|
|
} else if (statement1->kind == StatementKind::skKeyword) {
|
|
|
|
if (statement2->kind != StatementKind::skKeyword)
|
|
|
|
return true;
|
|
|
|
else
|
|
|
|
return statement1->command < statement2->command;
|
|
|
|
} else if (statement2->kind == StatementKind::skKeyword) {
|
|
|
|
return false;
|
|
|
|
// Show stuff from local headers first
|
|
|
|
} else if (statement1->inSystemHeader && ! statement2->inSystemHeader) {
|
|
|
|
return true;
|
|
|
|
} else if (!statement1->inSystemHeader && statement2->inSystemHeader) {
|
|
|
|
return false;
|
|
|
|
// Show local statements first
|
|
|
|
} else if (statement1->scope != StatementScope::ssGlobal
|
2021-08-27 16:38:55 +08:00
|
|
|
&& statement2->scope == StatementScope::ssGlobal ) {
|
2021-08-24 15:05:10 +08:00
|
|
|
return true;
|
|
|
|
} else if (statement1->scope == StatementScope::ssGlobal
|
2021-08-27 16:38:55 +08:00
|
|
|
&& statement2->scope != StatementScope::ssGlobal ) {
|
2021-08-24 15:05:10 +08:00
|
|
|
return false;
|
|
|
|
// otherwise, sort by name
|
|
|
|
} else
|
|
|
|
return statement1->command < statement2->command;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeCompletionView::filterList(const QString &member)
|
|
|
|
{
|
|
|
|
QMutexLocker locker(&mMutex);
|
|
|
|
mCompletionStatementList.clear();
|
|
|
|
if (!mParser)
|
|
|
|
return;
|
|
|
|
if (!mParser->enabled())
|
|
|
|
return;
|
|
|
|
//we don't need to freeze here since we use smart pointers
|
|
|
|
// and data have been retrieved from the parser
|
|
|
|
// if (!mParser->freeze())
|
|
|
|
// return;
|
|
|
|
// {
|
|
|
|
// auto action = finally([this]{
|
|
|
|
// mParser->unFreeze();
|
|
|
|
// });
|
|
|
|
// if (mParserSerialId!=mParser->serialId()) {
|
|
|
|
// return;
|
|
|
|
// }
|
|
|
|
|
2021-08-25 00:20:07 +08:00
|
|
|
mCompletionStatementList.clear();
|
2021-08-24 15:05:10 +08:00
|
|
|
if (!member.isEmpty()) { // filter
|
2021-08-25 00:20:07 +08:00
|
|
|
mCompletionStatementList.reserve(mFullCompletionStatementList.size());
|
2021-08-24 15:05:10 +08:00
|
|
|
for (PStatement statement:mFullCompletionStatementList) {
|
|
|
|
Qt::CaseSensitivity cs = (mIgnoreCase?
|
|
|
|
Qt::CaseInsensitive:
|
|
|
|
Qt::CaseSensitive);
|
|
|
|
if (statement->command.startsWith(member, cs))
|
2021-08-25 00:20:07 +08:00
|
|
|
mCompletionStatementList.append(statement);
|
2021-08-24 15:05:10 +08:00
|
|
|
}
|
|
|
|
} else
|
2021-08-25 00:20:07 +08:00
|
|
|
mCompletionStatementList.append(mFullCompletionStatementList);
|
2021-08-24 15:05:10 +08:00
|
|
|
if (mRecordUsage) {
|
|
|
|
int topCount = 0;
|
|
|
|
int secondCount = 0;
|
|
|
|
int thirdCount = 0;
|
|
|
|
int usageCount;
|
2021-08-25 00:20:07 +08:00
|
|
|
for (PStatement statement:mCompletionStatementList) {
|
2021-08-24 15:05:10 +08:00
|
|
|
if (statement->usageCount == 0) {
|
|
|
|
usageCount = mSymbolUsage.value(statement->fullName,0);
|
|
|
|
if (usageCount == 0)
|
|
|
|
continue;
|
|
|
|
statement->usageCount = usageCount;
|
|
|
|
} else
|
|
|
|
usageCount = statement->usageCount;
|
|
|
|
if (usageCount>topCount) {
|
|
|
|
thirdCount = secondCount;
|
|
|
|
secondCount = topCount;
|
|
|
|
topCount = usageCount;
|
|
|
|
} else if (usageCount == topCount) {
|
|
|
|
continue;
|
|
|
|
} else if (usageCount > secondCount) {
|
|
|
|
thirdCount = secondCount;
|
|
|
|
secondCount = usageCount;
|
|
|
|
} else if (usageCount == secondCount) {
|
|
|
|
continue;
|
|
|
|
} else if (usageCount>thirdCount) {
|
|
|
|
thirdCount = usageCount;
|
|
|
|
}
|
|
|
|
}
|
2021-08-25 00:20:07 +08:00
|
|
|
for (PStatement statement:mCompletionStatementList) {
|
2021-08-24 15:05:10 +08:00
|
|
|
if (statement->usageCount == 0) {
|
|
|
|
statement->freqTop = 0;
|
|
|
|
} else if (statement->usageCount == topCount) {
|
|
|
|
statement->freqTop = 30;
|
|
|
|
} else if (statement->usageCount == secondCount) {
|
|
|
|
statement->freqTop = 20;
|
|
|
|
} else if (statement->usageCount == thirdCount) {
|
|
|
|
statement->freqTop = 10;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (mSortByScope) {
|
2021-08-25 00:20:07 +08:00
|
|
|
std::sort(mCompletionStatementList.begin(),
|
|
|
|
mCompletionStatementList.end(),
|
|
|
|
sortByScopeWithUsageComparator);
|
2021-08-24 15:05:10 +08:00
|
|
|
} else {
|
2021-08-25 00:20:07 +08:00
|
|
|
std::sort(mCompletionStatementList.begin(),
|
|
|
|
mCompletionStatementList.end(),
|
|
|
|
sortWithUsageComparator);
|
2021-08-24 15:05:10 +08:00
|
|
|
}
|
|
|
|
} else if (mSortByScope) {
|
2021-08-25 00:20:07 +08:00
|
|
|
std::sort(mCompletionStatementList.begin(),
|
|
|
|
mCompletionStatementList.end(),
|
|
|
|
sortByScopeComparator);
|
2021-08-24 15:05:10 +08:00
|
|
|
} else {
|
2021-08-25 00:20:07 +08:00
|
|
|
std::sort(mCompletionStatementList.begin(),
|
|
|
|
mCompletionStatementList.end(),
|
|
|
|
defaultComparator);
|
|
|
|
}
|
|
|
|
// }
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeCompletionView::getCompletionFor(const QString &fileName, const QString &phrase, int line)
|
|
|
|
{
|
|
|
|
if(!mParser)
|
|
|
|
return;
|
|
|
|
if (!mParser->enabled())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!mParser->freeze())
|
|
|
|
return;
|
|
|
|
{
|
|
|
|
auto action = finally([this]{
|
|
|
|
mParser->unFreeze();
|
|
|
|
});
|
|
|
|
|
|
|
|
//C++ preprocessor directives
|
|
|
|
if (phrase.startsWith('#')) {
|
|
|
|
if (mShowKeywords) {
|
|
|
|
for (QString keyword:CppDirectives) {
|
|
|
|
PStatement statement = std::make_shared<Statement>();
|
|
|
|
statement->command = keyword;
|
2021-08-27 16:38:55 +08:00
|
|
|
statement->kind = StatementKind::skKeyword;
|
2021-08-25 00:20:07 +08:00
|
|
|
statement->fullName = keyword;
|
2021-08-27 16:38:55 +08:00
|
|
|
statement->usageCount = 0;
|
|
|
|
statement->freqTop = 0;
|
2021-08-25 00:20:07 +08:00
|
|
|
mFullCompletionStatementList.append(statement);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
//docstring tags (javadoc style)
|
|
|
|
if (phrase.startsWith('@')) {
|
|
|
|
if (mShowKeywords) {
|
|
|
|
for (QString keyword:JavadocTags) {
|
|
|
|
PStatement statement = std::make_shared<Statement>();
|
|
|
|
statement->command = keyword;
|
|
|
|
statement->kind = StatementKind::skKeyword;
|
|
|
|
statement->fullName = keyword;
|
2021-08-27 16:38:55 +08:00
|
|
|
statement->usageCount = 0;
|
|
|
|
statement->freqTop = 0;
|
2021-08-25 00:20:07 +08:00
|
|
|
mFullCompletionStatementList.append(statement);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Pulling off the same trick as in TCppParser.FindStatementOf, but ignore everything after last operator
|
|
|
|
int i = mParser->findLastOperator(phrase);
|
|
|
|
if (i < 0 ) { // don't have any scope prefix
|
|
|
|
|
|
|
|
if (mShowCodeIns) {
|
|
|
|
//add custom code templates
|
|
|
|
for (PCodeIns codeIn:mCodeInsList) {
|
|
|
|
PStatement statement = std::make_shared<Statement>();
|
|
|
|
statement->command = codeIn->prefix;
|
|
|
|
statement->kind = StatementKind::skUserCodeIn;
|
|
|
|
statement->fullName = codeIn->prefix;
|
2021-08-27 16:38:55 +08:00
|
|
|
statement->usageCount = 0;
|
|
|
|
statement->freqTop = 0;
|
2021-08-25 00:20:07 +08:00
|
|
|
mFullCompletionStatementList.append(statement);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mShowKeywords) {
|
|
|
|
//add keywords
|
|
|
|
if (mUseCppKeyword) {
|
|
|
|
for (QString keyword:CppKeywords.keys()) {
|
|
|
|
PStatement statement = std::make_shared<Statement>();
|
|
|
|
statement->command = keyword;
|
|
|
|
statement->kind = StatementKind::skKeyword;
|
|
|
|
statement->fullName = keyword;
|
2021-08-27 16:38:55 +08:00
|
|
|
statement->usageCount = 0;
|
|
|
|
statement->freqTop = 0;
|
2021-08-25 00:20:07 +08:00
|
|
|
mFullCompletionStatementList.append(statement);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (QString keyword:CKeywords) {
|
|
|
|
PStatement statement = std::make_shared<Statement>();
|
|
|
|
statement->command = keyword;
|
|
|
|
statement->kind = StatementKind::skKeyword;
|
|
|
|
statement->fullName = keyword;
|
2021-08-27 16:38:55 +08:00
|
|
|
statement->usageCount = 0;
|
|
|
|
statement->freqTop = 0;
|
2021-08-25 00:20:07 +08:00
|
|
|
mFullCompletionStatementList.append(statement);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
PStatement scopeStatement = mCurrentStatement;
|
|
|
|
// repeat until reach global
|
|
|
|
while (scopeStatement) {
|
|
|
|
//add members of current scope that not added before
|
|
|
|
if (scopeStatement->kind == StatementKind::skClass) {
|
|
|
|
addChildren(scopeStatement, fileName, -1);
|
|
|
|
} else {
|
|
|
|
addChildren(scopeStatement, fileName, line);
|
|
|
|
}
|
|
|
|
|
|
|
|
// add members of all usings (in current scope ) and not added before
|
|
|
|
for (QString namespaceName:scopeStatement->usingList) {
|
|
|
|
PStatementList namespaceStatementsList =
|
|
|
|
mParser->findNamespace(namespaceName);
|
|
|
|
if (!namespaceStatementsList)
|
|
|
|
continue;
|
|
|
|
for (PStatement namespaceStatement:*namespaceStatementsList) {
|
|
|
|
addChildren(namespaceStatement, fileName, line);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
scopeStatement=scopeStatement->parentScope.lock();
|
|
|
|
}
|
|
|
|
|
|
|
|
// add all global members and not added before
|
|
|
|
addChildren(nullptr, fileName, line);
|
|
|
|
|
|
|
|
// add members of all fusings
|
|
|
|
mUsings = mParser->getFileUsings(fileName);
|
|
|
|
for (QString namespaceName:mUsings) {
|
|
|
|
PStatementList namespaceStatementsList =
|
|
|
|
mParser->findNamespace(namespaceName);
|
|
|
|
if (!namespaceStatementsList)
|
|
|
|
continue;
|
|
|
|
for (PStatement namespaceStatement:*namespaceStatementsList) {
|
|
|
|
addChildren(namespaceStatement, fileName, line);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else { //we are in some statement's scope
|
|
|
|
MemberOperatorType opType=getOperatorType(phrase,i);
|
|
|
|
QString scopeName = phrase.mid(0,i);
|
|
|
|
if (opType == MemberOperatorType::otDColon) {
|
|
|
|
if (scopeName.isEmpty()) {
|
|
|
|
// start with '::', we only find in global
|
|
|
|
// add all global members and not added before
|
|
|
|
addChildren(nullptr, fileName, line);
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
//assume the scope its a namespace
|
|
|
|
PStatementList namespaceStatementsList =
|
|
|
|
mParser->findNamespace(scopeName);
|
|
|
|
if (namespaceStatementsList) {
|
|
|
|
for (PStatement namespaceStatement:*namespaceStatementsList) {
|
|
|
|
addChildren(namespaceStatement, fileName, line);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
//namespace not found let's go on
|
|
|
|
}
|
|
|
|
}
|
|
|
|
PStatement parentTypeStatement;
|
|
|
|
PStatement statement = mParser->findStatementOf(
|
|
|
|
fileName, scopeName,mCurrentStatement,parentTypeStatement);
|
|
|
|
if (!statement)
|
|
|
|
return;
|
|
|
|
// find the most inner scope statement that has a name (not a block)
|
|
|
|
PStatement scopeTypeStatement = mCurrentStatement;
|
|
|
|
while (scopeTypeStatement && !isScopeTypeKind(scopeTypeStatement->kind)) {
|
|
|
|
scopeTypeStatement = scopeTypeStatement->parentScope.lock();
|
|
|
|
}
|
|
|
|
if (
|
|
|
|
(opType == MemberOperatorType::otArrow
|
|
|
|
|| opType == MemberOperatorType::otDot)
|
|
|
|
&& (
|
|
|
|
statement->kind == StatementKind::skVariable
|
|
|
|
|| statement->kind == StatementKind::skParameter
|
|
|
|
|| statement->kind == StatementKind::skFunction)
|
|
|
|
) {
|
|
|
|
// Get type statement of current (scope) statement
|
|
|
|
PStatement classTypeStatement;
|
|
|
|
PStatement parentScope = statement->parentScope.lock();
|
|
|
|
if ((statement->kind == StatementKind::skFunction)
|
|
|
|
&& parentScope
|
|
|
|
&& STLContainers.contains(parentScope->fullName)
|
|
|
|
&& STLElementMethods.contains(statement->command)){
|
|
|
|
// it's an element method of STL container
|
|
|
|
// we must find the type in the template parameter
|
|
|
|
|
|
|
|
// get the function's owner variable's definition
|
|
|
|
int lastI = mParser->findLastOperator(scopeName);
|
|
|
|
QString lastScopeName = scopeName.mid(0,lastI);
|
|
|
|
PStatement lastScopeStatement =
|
|
|
|
mParser->findStatementOf(
|
|
|
|
fileName, lastScopeName,
|
|
|
|
mCurrentStatement,parentTypeStatement);
|
|
|
|
if (!lastScopeStatement)
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
QString typeName =
|
|
|
|
mParser->findFirstTemplateParamOf(
|
|
|
|
fileName,lastScopeStatement->type,
|
|
|
|
lastScopeStatement->parentScope.lock());
|
|
|
|
classTypeStatement = mParser->findTypeDefinitionOf(
|
|
|
|
fileName, typeName,
|
|
|
|
lastScopeStatement->parentScope.lock());
|
|
|
|
} else
|
|
|
|
classTypeStatement=mParser->findTypeDefinitionOf(
|
|
|
|
fileName, statement->type,parentTypeStatement);
|
|
|
|
|
|
|
|
if (!classTypeStatement)
|
|
|
|
return;
|
|
|
|
//is a smart pointer
|
|
|
|
if (STLPointers.contains(classTypeStatement->fullName)
|
|
|
|
&& (opType == MemberOperatorType::otArrow)) {
|
|
|
|
QString typeName= mParser->findFirstTemplateParamOf(
|
|
|
|
fileName,
|
|
|
|
statement->type,
|
|
|
|
parentScope);
|
|
|
|
classTypeStatement = mParser->findTypeDefinitionOf(
|
|
|
|
fileName,
|
|
|
|
typeName,
|
|
|
|
parentScope);
|
|
|
|
if (!classTypeStatement)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
//is a stl container operator[]
|
|
|
|
if (STLContainers.contains(classTypeStatement->fullName)
|
|
|
|
&& scopeName.endsWith(']')) {
|
|
|
|
QString typeName= mParser->findFirstTemplateParamOf(
|
|
|
|
fileName,
|
|
|
|
statement->type,
|
|
|
|
parentScope);
|
|
|
|
classTypeStatement = mParser->findTypeDefinitionOf(
|
|
|
|
fileName,
|
|
|
|
typeName,
|
|
|
|
parentScope);
|
|
|
|
if (!classTypeStatement)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!isIncluded(classTypeStatement->fileName) &&
|
|
|
|
!isIncluded(classTypeStatement->definitionFileName))
|
|
|
|
return;
|
|
|
|
if ((classTypeStatement == scopeTypeStatement) || (statement->command == "this")) {
|
|
|
|
//we can use all members
|
|
|
|
addChildren(classTypeStatement,fileName,-1);
|
|
|
|
} else { // we can only use public members
|
|
|
|
const StatementMap& children = mParser->statementList().childrenStatements(classTypeStatement);
|
|
|
|
if (children.isEmpty())
|
|
|
|
return;
|
|
|
|
for (PStatement childStatement:children) {
|
|
|
|
if ((childStatement->classScope==StatementClassScope::scsPublic)
|
|
|
|
&& !(
|
|
|
|
childStatement->kind == StatementKind::skConstructor
|
|
|
|
|| childStatement->kind == StatementKind::skDestructor)
|
|
|
|
&& !mAddedStatements.contains(childStatement->command)) {
|
|
|
|
addStatement(childStatement,fileName,-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//todo friend
|
|
|
|
} else if ((opType == MemberOperatorType::otDColon)
|
|
|
|
&& (statement->kind == StatementKind::skEnumType)
|
|
|
|
&& (statement->kind == StatementKind::skEnumClassType)) {
|
|
|
|
//we can add all child enum definess
|
|
|
|
PStatement classTypeStatement = statement;
|
|
|
|
if (!isIncluded(classTypeStatement->fileName) &&
|
|
|
|
!isIncluded(classTypeStatement->definitionFileName))
|
|
|
|
return;
|
|
|
|
const StatementMap& children =
|
|
|
|
mParser->statementList().childrenStatements(classTypeStatement);
|
|
|
|
for (PStatement child:children) {
|
|
|
|
addStatement(child,fileName,line);
|
|
|
|
}
|
|
|
|
} else if ((opType == MemberOperatorType::otDColon)
|
|
|
|
&& (statement->kind == StatementKind::skClass)) {
|
|
|
|
PStatement classTypeStatement = statement;
|
|
|
|
if (!isIncluded(classTypeStatement->fileName) &&
|
|
|
|
!isIncluded(classTypeStatement->definitionFileName))
|
|
|
|
return;
|
|
|
|
if (classTypeStatement == scopeTypeStatement) {
|
|
|
|
//we can use all static members
|
|
|
|
const StatementMap& children =
|
|
|
|
mParser->statementList().childrenStatements(classTypeStatement);
|
|
|
|
for (PStatement childStatement: children) {
|
|
|
|
if (
|
|
|
|
(childStatement->isStatic)
|
|
|
|
|| (childStatement->kind == StatementKind::skTypedef
|
|
|
|
|| childStatement->kind == StatementKind::skClass
|
|
|
|
|| childStatement->kind == StatementKind::skEnum
|
|
|
|
|| childStatement->kind == StatementKind::skEnumClassType
|
|
|
|
|| childStatement->kind == StatementKind::skEnumType
|
|
|
|
)) {
|
|
|
|
addStatement(childStatement,fileName,-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// we can only use public static members
|
|
|
|
const StatementMap& children =
|
|
|
|
mParser->statementList().childrenStatements(classTypeStatement);
|
|
|
|
for (PStatement childStatement: children) {
|
|
|
|
if (
|
|
|
|
(childStatement->isStatic)
|
|
|
|
|| (childStatement->kind == StatementKind::skTypedef
|
|
|
|
|| childStatement->kind == StatementKind::skClass
|
|
|
|
|| childStatement->kind == StatementKind::skEnum
|
|
|
|
|| childStatement->kind == StatementKind::skEnumClassType
|
|
|
|
|| childStatement->kind == StatementKind::skEnumType
|
|
|
|
)) {
|
|
|
|
if (childStatement->classScope == StatementClassScope::scsPublic)
|
|
|
|
addStatement(childStatement,fileName,-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//todo friend
|
|
|
|
}
|
|
|
|
}
|
2021-08-24 15:05:10 +08:00
|
|
|
}
|
2021-08-25 00:20:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CodeCompletionView::isIncluded(const QString &fileName)
|
|
|
|
{
|
|
|
|
return mIncludedFiles.contains(fileName);
|
|
|
|
}
|
|
|
|
|
2021-08-26 17:48:23 +08:00
|
|
|
void CodeCompletionView::showEvent(QShowEvent *)
|
|
|
|
{
|
|
|
|
mListView->setFocus();
|
|
|
|
}
|
|
|
|
|
2021-08-25 23:53:35 +08:00
|
|
|
const PStatement &CodeCompletionView::currentStatement() const
|
|
|
|
{
|
|
|
|
return mCurrentStatement;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeCompletionView::setCurrentStatement(const PStatement &newCurrentStatement)
|
|
|
|
{
|
|
|
|
mCurrentStatement = newCurrentStatement;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CodeCompletionView::useCppKeyword() const
|
|
|
|
{
|
|
|
|
return mUseCppKeyword;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeCompletionView::setUseCppKeyword(bool newUseCppKeyword)
|
|
|
|
{
|
|
|
|
mUseCppKeyword = newUseCppKeyword;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CodeCompletionView::sortByScope() const
|
|
|
|
{
|
|
|
|
return mSortByScope;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeCompletionView::setSortByScope(bool newSortByScope)
|
|
|
|
{
|
|
|
|
mSortByScope = newSortByScope;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CodeCompletionView::ignoreCase() const
|
|
|
|
{
|
|
|
|
return mIgnoreCase;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeCompletionView::setIgnoreCase(bool newIgnoreCase)
|
|
|
|
{
|
|
|
|
mIgnoreCase = newIgnoreCase;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CodeCompletionView::showCodeIns() const
|
|
|
|
{
|
|
|
|
return mShowCodeIns;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeCompletionView::setShowCodeIns(bool newShowCodeIns)
|
|
|
|
{
|
|
|
|
mShowCodeIns = newShowCodeIns;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CodeCompletionView::showKeywords() const
|
|
|
|
{
|
|
|
|
return mShowKeywords;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeCompletionView::setShowKeywords(bool newShowKeywords)
|
|
|
|
{
|
|
|
|
mShowKeywords = newShowKeywords;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CodeCompletionView::recordUsage() const
|
|
|
|
{
|
|
|
|
return mRecordUsage;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeCompletionView::setRecordUsage(bool newRecordUsage)
|
|
|
|
{
|
|
|
|
mRecordUsage = newRecordUsage;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CodeCompletionView::onlyGlobals() const
|
|
|
|
{
|
|
|
|
return mOnlyGlobals;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeCompletionView::setOnlyGlobals(bool newOnlyGlobals)
|
|
|
|
{
|
|
|
|
mOnlyGlobals = newOnlyGlobals;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CodeCompletionView::showCount() const
|
|
|
|
{
|
|
|
|
return mShowCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeCompletionView::setShowCount(int newShowCount)
|
|
|
|
{
|
|
|
|
mShowCount = newShowCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
const PCppParser &CodeCompletionView::parser() const
|
|
|
|
{
|
|
|
|
return mParser;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeCompletionView::setParser(const PCppParser &newParser)
|
|
|
|
{
|
|
|
|
mParser = newParser;
|
|
|
|
}
|
|
|
|
|
2021-08-25 00:20:07 +08:00
|
|
|
void CodeCompletionView::hideEvent(QHideEvent *event)
|
|
|
|
{
|
|
|
|
QMutexLocker locker(&mMutex);
|
2021-08-25 08:48:33 +08:00
|
|
|
mListView->setKeypressedCallback(nullptr);
|
2021-08-25 00:20:07 +08:00
|
|
|
mCompletionStatementList.clear();
|
|
|
|
mFullCompletionStatementList.clear();
|
|
|
|
mIncludedFiles.clear();
|
|
|
|
mUsings.clear();
|
|
|
|
mAddedStatements.clear();
|
|
|
|
QWidget::hideEvent(event);
|
2021-08-24 15:05:10 +08:00
|
|
|
}
|
|
|
|
|
2021-08-27 08:52:20 +08:00
|
|
|
bool CodeCompletionView::event(QEvent *event)
|
|
|
|
{
|
2021-08-27 16:38:55 +08:00
|
|
|
bool result = QWidget::event(event);
|
2021-08-27 08:52:20 +08:00
|
|
|
if (event->type() == QEvent::FontChange) {
|
2021-08-27 16:38:55 +08:00
|
|
|
mListView->setFont(font());
|
2021-08-27 08:52:20 +08:00
|
|
|
}
|
2021-08-27 16:38:55 +08:00
|
|
|
return result;
|
2021-08-27 08:52:20 +08:00
|
|
|
}
|
|
|
|
|
2021-08-24 09:59:44 +08:00
|
|
|
CodeCompletionListView::CodeCompletionListView(QWidget *parent) : QListView(parent)
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeCompletionListView::keyPressEvent(QKeyEvent *event)
|
|
|
|
{
|
2021-08-27 08:52:20 +08:00
|
|
|
if (event->key() == Qt::Key_Up
|
|
|
|
|| event->key() == Qt::Key_Down) {
|
|
|
|
QListView::keyPressEvent(event);
|
|
|
|
return;
|
|
|
|
}
|
2021-08-25 08:48:33 +08:00
|
|
|
if (!mKeypressedCallback || !mKeypressedCallback(event)) {
|
2021-08-24 09:59:44 +08:00
|
|
|
QListView::keyPressEvent(event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const KeyPressedCallback &CodeCompletionListView::keypressedCallback() const
|
|
|
|
{
|
|
|
|
return mKeypressedCallback;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeCompletionListView::setKeypressedCallback(const KeyPressedCallback &newKeypressedCallback)
|
|
|
|
{
|
|
|
|
mKeypressedCallback = newKeypressedCallback;
|
2021-08-23 21:50:53 +08:00
|
|
|
}
|
2021-08-25 08:48:33 +08:00
|
|
|
|
|
|
|
CodeCompletionListModel::CodeCompletionListModel(StatementList *statements, QObject *parent):
|
|
|
|
QAbstractListModel(parent),
|
|
|
|
mStatements(statements)
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2021-08-25 23:53:35 +08:00
|
|
|
int CodeCompletionListModel::rowCount(const QModelIndex &) const
|
2021-08-25 08:48:33 +08:00
|
|
|
{
|
|
|
|
return mStatements->count();
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant CodeCompletionListModel::data(const QModelIndex &index, int role) const
|
|
|
|
{
|
|
|
|
if (!index.isValid())
|
|
|
|
return QVariant();
|
2021-08-26 17:48:23 +08:00
|
|
|
if (index.row()>=mStatements->count())
|
|
|
|
return QVariant();
|
|
|
|
|
2021-08-25 08:48:33 +08:00
|
|
|
if (role == Qt::DisplayRole) {
|
|
|
|
PStatement statement = mStatements->at(index.row());
|
|
|
|
return statement->command;
|
|
|
|
}
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeCompletionListModel::notifyUpdated()
|
|
|
|
{
|
|
|
|
beginResetModel();
|
|
|
|
endResetModel();
|
|
|
|
}
|