RedPanda-CPP/RedPandaIDE/widgets/codecompletionpopup.cpp

1450 lines
54 KiB
C++
Raw Normal View History

2021-12-26 23:18:28 +08:00
/*
* Copyright (C) 2020-2022 Roy Qu (royqh1979@gmail.com)
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
2021-08-29 00:48:23 +08:00
#include "codecompletionpopup.h"
2021-08-24 15:05:10 +08:00
#include "../utils.h"
#include "../mainwindow.h"
#include "../editor.h"
#include "../editorlist.h"
#include "../symbolusagemanager.h"
#include "../colorscheme.h"
#include "../iconsmanager.h"
#include "../settings.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-28 09:01:40 +08:00
#include <QApplication>
#include <QPainter>
2021-08-24 09:59:44 +08:00
2021-08-29 00:48:23 +08:00
CodeCompletionPopup::CodeCompletionPopup(QWidget *parent) :
2022-01-04 16:50:54 +08:00
QWidget(parent),
mMutex()
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);
mDelegate = new CodeCompletionListItemDelegate(mModel,this);
QItemSelectionModel *m=mListView->selectionModel();
2021-08-25 08:48:33 +08:00
mListView->setModel(mModel);
delete m;
mListView->setItemDelegate(mDelegate);
2021-08-24 09:59:44 +08:00
setLayout(new QVBoxLayout());
layout()->addWidget(mListView);
layout()->setMargin(0);
2023-10-18 21:52:58 +08:00
mListView->setFocus();
2021-08-25 00:20:07 +08:00
mShowKeywords=true;
2021-08-27 16:38:55 +08:00
mRecordUsage = false;
mSortByScope = true;
2021-08-25 00:20:07 +08:00
mShowCount = 1000;
2021-09-30 20:10:48 +08:00
mShowCodeSnippets = true;
2021-08-25 00:20:07 +08:00
mIgnoreCase = false;
mHideSymbolsStartWithTwoUnderline = false;
mHideSymbolsStartWithUnderline = false;
2021-08-23 21:50:53 +08:00
}
2021-08-29 00:48:23 +08:00
CodeCompletionPopup::~CodeCompletionPopup()
2021-08-23 21:50:53 +08:00
{
2021-08-25 08:48:33 +08:00
delete mListView;
delete mModel;
2021-08-24 09:59:44 +08:00
}
2021-08-29 00:48:23 +08:00
void CodeCompletionPopup::setKeypressedCallback(const KeyPressedCallback &newKeypressedCallback)
2021-08-24 09:59:44 +08:00
{
mListView->setKeypressedCallback(newKeypressedCallback);
}
2021-12-04 10:02:07 +08:00
void CodeCompletionPopup::prepareSearch(
const QString &preWord,
const QStringList &ownerExpression,
const QString& memberOperator,
const QStringList& memberExpression,
const QString &filename,
int line,
CodeCompletionType type,
const QSet<QString>& customKeywords)
2021-12-03 11:40:05 +08:00
{
QMutexLocker locker(&mMutex);
if (!isEnabled())
return;
QCursor oldCursor = cursor();
setCursor(Qt::CursorShape::WaitCursor);
2021-12-04 10:02:07 +08:00
mMemberPhrase = memberExpression.join("");
mMemberOperator = memberOperator;
switch(type) {
case CodeCompletionType::ComplexKeyword:
getCompletionListForComplexKeyword(preWord);
break;
case CodeCompletionType::Types:
mIncludedFiles = mParser->getIncludedFiles(filename);
getCompletionListForTypes(preWord,filename,line);
break;
case CodeCompletionType::FunctionWithoutDefinition:
mIncludedFiles = mParser->getIncludedFiles(filename);
getCompletionForFunctionWithoutDefinition(preWord, ownerExpression,memberOperator,memberExpression, filename,line);
break;
case CodeCompletionType::Namespaces:
mIncludedFiles = mParser->getIncludedFiles(filename);
getCompletionListForNamespaces(preWord,filename,line);
break;
case CodeCompletionType::KeywordsOnly:
mIncludedFiles.clear();
getKeywordCompletionFor(customKeywords);
break;
case CodeCompletionType::Macros:
mIncludedFiles = mParser->getIncludedFiles(filename);
getMacroCompletionList(filename, line);
break;
default:
mIncludedFiles = mParser->getIncludedFiles(filename);
getCompletionFor(ownerExpression,memberOperator,memberExpression, filename,line, customKeywords);
2021-12-03 11:40:05 +08:00
}
setCursor(oldCursor);
}
2021-12-04 14:53:21 +08:00
bool CodeCompletionPopup::search(const QString &memberPhrase, bool autoHideOnSingleResult)
2021-08-25 00:20:07 +08:00
{
QMutexLocker locker(&mMutex);
2021-12-04 14:53:21 +08:00
mMemberPhrase = memberPhrase;
2021-08-25 00:20:07 +08:00
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);
// filter fFullCompletionStatementList to fCompletionStatementList
2021-12-04 14:53:21 +08:00
filterList(memberPhrase);
//if can't find a destructor, maybe '~' is only an operator
2021-12-04 10:02:07 +08:00
// if (mCompletionStatementList.isEmpty() && phrase.startsWith('~')) {
// symbol = phrase.mid(1);
// 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()) {
QString schemaName = pSettings->editor().colorScheme();
PColorSchemeItem item = pColorManager->getItem(schemaName, COLOR_SCHEME_ACTIVE_LINE);
if (item)
mDelegate->setCurrentSelectionColor(item->background());
else
mDelegate->setCurrentSelectionColor(palette().highlight().color());
item = pColorManager->getItem(schemaName, COLOR_SCHEME_TEXT);
if (item)
mDelegate->setNormalColor(item->foreground());
else
mDelegate->setNormalColor(palette().color(QPalette::Text));
item = pColorManager->getItem(schemaName, SYNS_AttrReserveWord_Type);
if (item)
mDelegate->setMatchedColor(item->foreground());
else
mDelegate->setMatchedColor(palette().color(QPalette::HighlightedText));
mListView->setCurrentIndex(mModel->index(0,0));
2021-08-25 00:20:07 +08:00
// 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
2021-12-04 14:53:21 +08:00
|| (memberPhrase == mCompletionStatementList.front()->command)) {
2021-08-25 00:20:07 +08:00
return true;
}
} else {
hide();
}
2021-08-25 08:48:33 +08:00
return false;
2021-08-25 00:20:07 +08:00
}
2021-08-29 00:48:23 +08:00
PStatement CodeCompletionPopup::selectedStatement()
2021-08-26 17:48:23 +08:00
{
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();
}
void CodeCompletionPopup::addChildren(const PStatement& scopeStatement,
const QString &fileName,
int line,
bool onlyTypes)
2021-08-24 15:05:10 +08:00
{
if (scopeStatement && !isIncluded(scopeStatement->fileName)
&& !isIncluded(scopeStatement->definitionFileName))
return;
const StatementMap& children = mParser->statementList().childrenStatements(scopeStatement);
if (children.isEmpty())
return;
if (onlyTypes) {
if (!scopeStatement) { //Global scope
for (const PStatement& childStatement: children) {
if (!isTypeKind(childStatement->kind))
continue;
if (childStatement->fileName.isEmpty()) {
// hard defines
addStatement(childStatement,fileName,-1);
} else if (
isIncluded(childStatement->fileName)
|| isIncluded(childStatement->definitionFileName)
) {
//we must check if the statement is included by the file
addStatement(childStatement,fileName,line);
}
}
} else {
for (const PStatement& childStatement: children) {
if (!isTypeKind(childStatement->kind))
continue;
2021-08-24 15:05:10 +08:00
addStatement(childStatement,fileName,line);
}
}
} else {
if (!scopeStatement) { //Global scope
for (const PStatement& childStatement: children) {
if (childStatement->fileName.isEmpty()) {
// hard defines
addStatement(childStatement,fileName,-1);
} else if (
isIncluded(childStatement->fileName)
|| isIncluded(childStatement->definitionFileName)
) {
//we must check if the statement is included by the file
addStatement(childStatement,fileName,line);
}
}
} else {
for (const PStatement& childStatement: children) {
addStatement(childStatement,fileName,line);
}
2021-08-24 15:05:10 +08:00
}
2021-08-24 15:05:10 +08:00
}
}
2022-11-28 11:28:02 +08:00
void CodeCompletionPopup::addFunctionWithoutDefinitionChildren(const 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;
for (const PStatement& childStatement: children) {
2022-11-16 10:29:20 +08:00
if (childStatement->inSystemHeader())
continue;
if (childStatement->fileName.isEmpty()) {
// hard defines, do nothing
continue;
}
switch(childStatement->kind) {
2024-04-03 10:37:29 +08:00
case StatementKind::Constructor:
case StatementKind::Function:
case StatementKind::Destructor:
2022-11-16 10:29:20 +08:00
if (!childStatement->hasDefinition())
addStatement(childStatement,fileName,line);
break;
2024-04-03 10:37:29 +08:00
case StatementKind::Class:
case StatementKind::Namespace:
if (isIncluded(childStatement->fileName))
addStatement(childStatement,fileName,line);
break;
default:
break;
}
}
}
2022-11-28 11:28:02 +08:00
void CodeCompletionPopup::addStatement(const PStatement& statement, const QString &fileName, int line)
2021-08-24 15:05:10 +08:00
{
2021-08-25 00:20:07 +08:00
if (mAddedStatements.contains(statement->command))
return;
2024-04-03 10:37:29 +08:00
if (statement->kind == StatementKind::Constructor
|| statement->kind == StatementKind::Destructor
|| statement->kind == StatementKind::Block
|| statement->kind == StatementKind::Lambda
|| statement->properties.testFlag(StatementProperty::OperatorOverloading)
|| statement->properties.testFlag(StatementProperty::DummyStatement)
)
2022-10-28 09:47:34 +08:00
return;
2021-08-24 15:05:10 +08:00
if ((line!=-1)
&& (line < statement->line)
&& (fileName == statement->fileName))
return;
mAddedStatements.insert(statement->command);
2024-04-03 10:37:29 +08:00
if (statement->kind == StatementKind::UserCodeSnippet || !statement->command.contains("<"))
mFullCompletionStatementList.append(statement);
2021-08-24 15:05:10 +08:00
}
static bool nameComparator(PStatement statement1,PStatement statement2) {
return statement1->command < statement2->command;
}
static bool defaultComparator(PStatement statement1,PStatement statement2) {
if (statement1->matchPosSpan!=statement2->matchPosSpan)
return statement1->matchPosSpan < statement2->matchPosSpan;
if (statement1->firstMatchLength != statement2->firstMatchLength)
return statement1->firstMatchLength > statement2->firstMatchLength;
if (statement1->matchPosTotal != statement2->matchPosTotal)
return statement1->matchPosTotal < statement2->matchPosTotal;
if (statement1->caseMatched != statement2->caseMatched)
return statement1->caseMatched > statement2->caseMatched;
2021-08-24 15:05:10 +08:00
// Show user template first
2024-04-03 10:37:29 +08:00
if (statement1->kind == StatementKind::UserCodeSnippet) {
if (statement2->kind != StatementKind::UserCodeSnippet)
2021-08-24 15:05:10 +08:00
return true;
else
return statement1->command < statement2->command;
2024-04-03 10:37:29 +08:00
} else if (statement2->kind == StatementKind::UserCodeSnippet) {
2021-08-24 15:05:10 +08:00
return false;
// show keywords first
2024-04-03 10:37:29 +08:00
} else if ((statement1->kind == StatementKind::Keyword)
&& (statement2->kind != StatementKind::Keyword)) {
2021-08-24 15:05:10 +08:00
return true;
2024-04-03 10:37:29 +08:00
} else if ((statement1->kind != StatementKind::Keyword)
&& (statement2->kind == StatementKind::Keyword)) {
2021-08-24 15:05:10 +08:00
return false;
} else
return nameComparator(statement1,statement2);
2021-08-24 15:05:10 +08:00
}
static bool sortByScopeComparator(PStatement statement1,PStatement statement2){
if (statement1->matchPosSpan!=statement2->matchPosSpan)
return statement1->matchPosSpan < statement2->matchPosSpan;
if (statement1->firstMatchLength != statement2->firstMatchLength)
return statement1->firstMatchLength > statement2->firstMatchLength;
if (statement1->matchPosTotal != statement2->matchPosTotal)
return statement1->matchPosTotal < statement2->matchPosTotal;
if (statement1->caseMatched != statement2->caseMatched)
return statement1->caseMatched > statement2->caseMatched;
2021-08-24 15:05:10 +08:00
// Show user template first
2024-04-03 10:37:29 +08:00
if (statement1->kind == StatementKind::UserCodeSnippet) {
if (statement2->kind != StatementKind::UserCodeSnippet)
2021-08-24 15:05:10 +08:00
return true;
else
return statement1->command < statement2->command;
2024-04-03 10:37:29 +08:00
} else if (statement2->kind == StatementKind::UserCodeSnippet) {
2021-08-24 15:05:10 +08:00
return false;
// show non-system defines before keyword
2024-04-03 10:37:29 +08:00
} else if (statement1->kind == StatementKind::Keyword) {
if (statement2->kind != StatementKind::Keyword) {
//s1 keyword / s2 system defines, s1 < s2, should return true
//s1 keyword / s2 not system defines, s2 < s1, should return false;
2022-11-16 10:29:20 +08:00
return statement2->inSystemHeader();
} else
2021-08-24 15:05:10 +08:00
return statement1->command < statement2->command;
2024-04-03 10:37:29 +08:00
} else if (statement2->kind == StatementKind::Keyword) {
//s1 system defines / s2 keyword, s2 < s1, should return false;
//s1 not system defines / s2 keyword, s1 < s2, should return true;
2022-11-16 10:29:20 +08:00
return (!statement1->inSystemHeader());
}
// Show stuff from local headers first
2022-11-16 10:29:20 +08:00
if (statement1->inSystemHeader() != statement2->inSystemHeader())
return !(statement1->inSystemHeader());
2021-08-24 15:05:10 +08:00
// Show local statements first
2022-11-01 09:02:17 +08:00
if (statement1->scope != StatementScope::Global
&& statement2->scope == StatementScope::Global ) {
2021-08-24 15:05:10 +08:00
return true;
2022-11-01 09:02:17 +08:00
} else if (statement1->scope == StatementScope::Global
&& statement2->scope != StatementScope::Global ) {
2021-08-24 15:05:10 +08:00
return false;
} else
return nameComparator(statement1,statement2);
2021-08-24 15:05:10 +08:00
}
static bool sortWithUsageComparator(PStatement statement1,PStatement statement2) {
if (statement1->matchPosSpan!=statement2->matchPosSpan)
return statement1->matchPosSpan < statement2->matchPosSpan;
if (statement1->firstMatchLength != statement2->firstMatchLength)
return statement1->firstMatchLength > statement2->firstMatchLength;
if (statement1->matchPosTotal != statement2->matchPosTotal)
return statement1->matchPosTotal < statement2->matchPosTotal;
if (statement1->caseMatched != statement2->caseMatched)
return statement1->caseMatched > statement2->caseMatched;
2021-08-24 15:05:10 +08:00
// Show user template first
2024-04-03 10:37:29 +08:00
if (statement1->kind == StatementKind::UserCodeSnippet) {
if (statement2->kind != StatementKind::UserCodeSnippet)
2021-08-24 15:05:10 +08:00
return true;
else
return statement1->command < statement2->command;
2024-04-03 10:37:29 +08:00
} else if (statement2->kind == StatementKind::UserCodeSnippet) {
2021-08-24 15:05:10 +08:00
return false;
//show most freq first
}
if (statement1->usageCount != statement2->usageCount)
return statement1->usageCount > statement2->usageCount;
2024-04-03 10:37:29 +08:00
if ((statement1->kind != StatementKind::Keyword)
&& (statement2->kind == StatementKind::Keyword)) {
return true;
2024-04-03 10:37:29 +08:00
} else if ((statement1->kind == StatementKind::Keyword)
&& (statement2->kind != StatementKind::Keyword)) {
2021-08-24 15:05:10 +08:00
return false;
} else
return nameComparator(statement1,statement2);
2021-08-24 15:05:10 +08:00
}
static bool sortByScopeWithUsageComparator(PStatement statement1,PStatement statement2){
if (statement1->matchPosSpan!=statement2->matchPosSpan)
return statement1->matchPosSpan < statement2->matchPosSpan;
if (statement1->firstMatchLength != statement2->firstMatchLength)
return statement1->firstMatchLength > statement2->firstMatchLength;
if (statement1->matchPosTotal != statement2->matchPosTotal)
return statement1->matchPosTotal < statement2->matchPosTotal;
if (statement1->caseMatched != statement2->caseMatched)
return statement1->caseMatched > statement2->caseMatched;
2021-08-24 15:05:10 +08:00
// Show user template first
2024-04-03 10:37:29 +08:00
if (statement1->kind == StatementKind::UserCodeSnippet) {
if (statement2->kind != StatementKind::UserCodeSnippet)
2021-08-24 15:05:10 +08:00
return true;
else
return statement1->command < statement2->command;
2024-04-03 10:37:29 +08:00
} else if (statement2->kind == StatementKind::UserCodeSnippet) {
2021-08-24 15:05:10 +08:00
return false;
//show most freq first
}
if (statement1->usageCount != statement2->usageCount)
return statement1->usageCount > statement2->usageCount;
// show non-system defines before keyword
2024-04-03 10:37:29 +08:00
if (statement1->kind == StatementKind::Keyword) {
if (statement2->kind != StatementKind::Keyword) {
//s1 keyword / s2 system defines, s1 < s2, should return true
//s1 keyword / s2 not system defines, s2 < s1, should return false;
2022-11-16 10:29:20 +08:00
return statement2->inSystemHeader();
} else
2021-08-24 15:05:10 +08:00
return statement1->command < statement2->command;
2024-04-03 10:37:29 +08:00
} else if (statement2->kind == StatementKind::Keyword) {
//s1 system defines / s2 keyword, s2 < s1, should return false;
//s1 not system defines / s2 keyword, s1 < s2, should return true;
2022-11-16 10:29:20 +08:00
return (!statement1->inSystemHeader());
}
// Show stuff from local headers first
2022-11-16 10:29:20 +08:00
if (statement1->inSystemHeader() != statement2->inSystemHeader())
return !(statement1->inSystemHeader());
2021-08-24 15:05:10 +08:00
// Show local statements first
2022-11-01 09:02:17 +08:00
if (statement1->scope != StatementScope::Global
&& statement2->scope == StatementScope::Global ) {
2021-08-24 15:05:10 +08:00
return true;
2022-11-01 09:02:17 +08:00
} else if (statement1->scope == StatementScope::Global
&& statement2->scope != StatementScope::Global ) {
2021-08-24 15:05:10 +08:00
return false;
} else
return nameComparator(statement1,statement2);
2021-08-24 15:05:10 +08:00
}
2021-08-29 00:48:23 +08:00
void CodeCompletionPopup::filterList(const QString &member)
2021-08-24 15:05:10 +08:00
{
QMutexLocker locker(&mMutex);
mCompletionStatementList.clear();
// if (!mParser)
// return;
// if (!mParser->enabled())
// return;
2021-08-24 15:05:10 +08:00
//we don't need to freeze here since we use smart pointers
// and data have been retrieved from the parser
2021-08-25 00:20:07 +08:00
mCompletionStatementList.clear();
mCompletionStatementList.reserve(mFullCompletionStatementList.size());
bool hideSymbolsTwoUnderline = mHideSymbolsStartWithTwoUnderline && !member.startsWith("__") ;
bool hideSymbolsUnderline = mHideSymbolsStartWithUnderline && !member.startsWith("_") ;
int len = member.length();
foreach (const PStatement& statement, mFullCompletionStatementList) {
int matched = 0;
int caseMatched = 0;
QString command = statement->command;
int pos = 0;
int lastPos = -10;
int totalPos = 0;
statement->matchPositions.clear();
if (hideSymbolsTwoUnderline && statement->command.startsWith("__")) {
2022-10-28 09:47:34 +08:00
continue;
} else if (hideSymbolsUnderline && statement->command.startsWith("_")) {
2022-10-28 09:47:34 +08:00
continue;
} else {
foreach (const QChar& ch, member) {
if (mIgnoreCase)
pos = command.indexOf(ch,pos,Qt::CaseInsensitive);
else
pos = command.indexOf(ch,pos,Qt::CaseSensitive);
if (pos<0) {
break;
}
if (pos == lastPos+1) {
statement->matchPositions.last()->end++;
} else {
PStatementMathPosition matchPosition=std::make_shared<StatementMatchPosition>();
matchPosition->start = pos;
matchPosition->end = pos+1;
statement->matchPositions.append(matchPosition);
}
if (ch==command[pos])
caseMatched++;
matched++;
totalPos += pos;
lastPos = pos;
pos+=1;
}
2021-08-24 15:05:10 +08:00
}
if (mIgnoreCase && matched== len) {
statement->caseMatched = caseMatched;
statement->matchPosTotal = totalPos;
if (member.length()>0) {
statement->firstMatchLength = statement->matchPositions.front()->end - statement->matchPositions.front()->start;
statement->matchPosSpan = statement->matchPositions.last()->end - statement->matchPositions.front()->start;
} else
statement->firstMatchLength = 0;
mCompletionStatementList.append(statement);
} else if (caseMatched == len) {
statement->caseMatched = caseMatched;
statement->matchPosTotal = totalPos;
if (member.length()>0) {
statement->firstMatchLength = statement->matchPositions.front()->end - statement->matchPositions.front()->start;
statement->matchPosSpan = statement->matchPositions.last()->end - statement->matchPositions.front()->start;
} else
statement->firstMatchLength = 0;
mCompletionStatementList.append(statement);
} else {
statement->matchPositions.clear();
statement->caseMatched = 0;
statement->matchPosTotal = 0;
statement->firstMatchLength = 0;
statement->matchPosSpan = 0;
}
}
2021-08-24 15:05:10 +08:00
if (mRecordUsage) {
int usageCount;
2021-08-29 00:48:23 +08:00
foreach (const PStatement& statement,mCompletionStatementList) {
if (statement->usageCount == -1) {
PSymbolUsage usage = pMainWindow->symbolUsageManager()->findUsage(statement->fullName);
if (usage) {
usageCount = usage->count;
} else {
usageCount = 0;
}
2021-08-24 15:05:10 +08:00
statement->usageCount = usageCount;
}
}
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 CodeCompletionPopup::getKeywordCompletionFor(const QSet<QString> &customKeywords)
{
//add keywords
if (!customKeywords.isEmpty()) {
foreach (const QString& keyword,customKeywords) {
addKeyword(keyword);
}
}
}
void CodeCompletionPopup::getMacroCompletionList(const QString &fileName, int line)
{
const StatementMap& statementMap = mParser->statementList().childrenStatements(nullptr);
foreach(const PStatement& statement, statementMap.values()) {
2024-04-03 10:37:29 +08:00
if (statement->kind==StatementKind::Preprocessor)
addStatement(statement,fileName,line);
}
}
2021-12-04 10:02:07 +08:00
void CodeCompletionPopup::getCompletionFor(
QStringList ownerExpression,
2021-12-04 10:02:07 +08:00
const QString& memberOperator,
const QStringList& memberExpression,
const QString &fileName,
int line,
const QSet<QString>& customKeywords)
2021-08-25 00:20:07 +08:00
{
if (memberOperator.isEmpty() && ownerExpression.isEmpty() && memberExpression.isEmpty())
return;
bool isLambdaReturnType = (
memberOperator=="->"
&& ownerExpression.startsWith("[")
&& ownerExpression.endsWith(")"));
if (memberOperator.isEmpty()) {
//C++ preprocessor directives
if (mMemberPhrase.startsWith('#')) {
if (mShowKeywords) {
foreach (const QString& keyword, CppDirectives) {
addKeyword(keyword);
}
}
return;
}
//docstring tags (javadoc style)
if (mMemberPhrase.startsWith('@')) {
if (mShowKeywords) {
foreach (const QString& keyword,JavadocTags) {
addKeyword(keyword);
}
}
return;
}
//the identifier to be completed is not a member of variable/class
if (mShowCodeSnippets) {
//add custom code templates
foreach (const PCodeSnippet& codeIn,mCodeSnippets) {
if (!codeIn->code.isEmpty()) {
PStatement statement = std::make_shared<Statement>();
statement->command = codeIn->prefix;
statement->value = codeIn->code;
2024-04-03 10:37:29 +08:00
statement->kind = StatementKind::UserCodeSnippet;
statement->fullName = codeIn->prefix;
statement->usageCount = 0;
mFullCompletionStatementList.append(statement);
}
}
}
if (mShowKeywords) {
//add keywords
if (!customKeywords.isEmpty()) {
foreach (const QString& keyword,customKeywords) {
addKeyword(keyword);
}
}
}
} else if (isLambdaReturnType) {
foreach (const QString& keyword,CppTypeKeywords) {
addKeyword(keyword);
}
}
if (!mParser || !mParser->enabled())
2021-12-03 20:08:18 +08:00
return;
if (!mParser->freeze())
return;
{
auto action = finally([this]{
mParser->unFreeze();
});
if (memberOperator.isEmpty() || isLambdaReturnType) {
2022-11-10 08:05:04 +08:00
PStatement scopeStatement = mCurrentScope;
2021-12-03 20:08:18 +08:00
// repeat until reach global
while (scopeStatement) {
//add members of current scope that not added before
2024-04-03 10:37:29 +08:00
if (scopeStatement->kind == StatementKind::Namespace) {
PStatementList namespaceStatementsList =
mParser->findNamespace(scopeStatement->fullName);
foreach (const PStatement& namespaceStatement,*namespaceStatementsList) {
addChildren(namespaceStatement, fileName, line, isLambdaReturnType);
}
2024-04-03 10:37:29 +08:00
} else if (scopeStatement->kind == StatementKind::Class) {
addChildren(scopeStatement, fileName, -1, isLambdaReturnType);
2021-12-03 20:08:18 +08:00
} else {
addChildren(scopeStatement, fileName, line, isLambdaReturnType);
2021-12-03 20:08:18 +08:00
}
// add members of all usings (in current scope ) and not added before
2021-08-29 00:48:23 +08:00
foreach (const QString& namespaceName,scopeStatement->usingList) {
2021-08-25 00:20:07 +08:00
PStatementList namespaceStatementsList =
mParser->findNamespace(namespaceName);
if (!namespaceStatementsList)
continue;
2021-08-29 00:48:23 +08:00
foreach (const PStatement& namespaceStatement,*namespaceStatementsList) {
addChildren(namespaceStatement, fileName, line, isLambdaReturnType);
2021-08-25 00:20:07 +08:00
}
}
2024-04-03 10:37:29 +08:00
if (scopeStatement->kind == StatementKind::Lambda) {
2024-03-23 09:03:20 +08:00
foreach (const QString& phrase, scopeStatement->lambdaCaptures) {
if (phrase=="&" || phrase == "=" || phrase =="this")
continue;
PStatement statement = mParser->findStatementOf(
scopeStatement->fileName,
phrase,scopeStatement->line);
if (statement)
addStatement(statement,scopeStatement->fileName, scopeStatement->line);
}
if (scopeStatement->lambdaCaptures.contains("&")
|| scopeStatement->lambdaCaptures.contains("=")) {
scopeStatement = scopeStatement->parentScope.lock();
} else if (scopeStatement->lambdaCaptures.contains("this")) {
do {
scopeStatement = scopeStatement->parentScope.lock();
2024-04-03 10:37:29 +08:00
} while (scopeStatement && scopeStatement->kind!=StatementKind::Class
&& scopeStatement->kind!=StatementKind::Namespace);
2024-03-23 09:13:07 +08:00
} else {
do {
scopeStatement = scopeStatement->parentScope.lock();
2024-04-03 10:37:29 +08:00
} while (scopeStatement && scopeStatement->kind!=StatementKind::Namespace);
2024-03-23 09:03:20 +08:00
}
2024-03-23 09:13:07 +08:00
continue;
2024-03-23 09:03:20 +08:00
}
2021-08-25 00:20:07 +08:00
scopeStatement=scopeStatement->parentScope.lock();
}
// add all global members and not added before
addChildren(nullptr, fileName, line, isLambdaReturnType);
2021-08-25 00:20:07 +08:00
// add members of all fusings
mUsings = mParser->getFileUsings(fileName);
2021-08-29 00:48:23 +08:00
foreach (const QString& namespaceName, mUsings) {
2021-08-25 00:20:07 +08:00
PStatementList namespaceStatementsList =
mParser->findNamespace(namespaceName);
if (!namespaceStatementsList)
continue;
2021-08-29 00:48:23 +08:00
foreach (const PStatement& namespaceStatement, *namespaceStatementsList) {
addChildren(namespaceStatement, fileName, line, isLambdaReturnType);
2021-08-25 00:20:07 +08:00
}
}
2021-12-03 20:24:49 +08:00
} else {
//the identifier to be completed is a member of variable/class
2021-12-04 10:02:07 +08:00
if (memberOperator == "::" && ownerExpression.isEmpty()) {
2021-12-03 20:24:49 +08:00
// start with '::', we only find in global
// add all global members and not added before
addChildren(nullptr, fileName, line);
return;
}
if (memberExpression.length()==2 && memberExpression.front()!="~")
return;
if (memberExpression.length()>2)
return;
2021-12-08 19:13:47 +08:00
2022-11-10 08:05:04 +08:00
PStatement scope = mCurrentScope;//the scope the expression in
2021-12-03 20:24:49 +08:00
PStatement parentTypeStatement;
2021-12-08 19:13:47 +08:00
// QString scopeName = ownerExpression.join("");
// PStatement ownerStatement = mParser->findStatementOf(
// fileName,
// scopeName,
// mCurrentStatement,
// parentTypeStatement);
PEvalStatement ownerStatement = mParser->evalExpression(fileName,
ownerExpression,
scope);
2024-03-28 15:30:10 +08:00
2021-12-08 19:13:47 +08:00
if(!ownerStatement || !ownerStatement->effectiveTypeStatement) {
2021-12-03 20:24:49 +08:00
return;
}
if (memberOperator == "::") {
2021-12-08 19:13:47 +08:00
if (ownerStatement->kind==EvalStatementKind::Namespace) {
2021-12-03 20:24:49 +08:00
//there might be many statements corresponding to one namespace;
2021-08-25 00:20:07 +08:00
PStatementList namespaceStatementsList =
2021-12-08 19:13:47 +08:00
mParser->findNamespace(ownerStatement->baseType);
2021-08-25 00:20:07 +08:00
if (namespaceStatementsList) {
2021-08-29 00:48:23 +08:00
foreach (const PStatement& namespaceStatement, *namespaceStatementsList) {
2021-08-25 00:20:07 +08:00
addChildren(namespaceStatement, fileName, line);
}
}
2021-12-03 20:24:49 +08:00
return;
2021-08-25 00:20:07 +08:00
}
}
2021-08-25 00:20:07 +08:00
// find the most inner scope statement that has a name (not a block)
2022-11-10 08:05:04 +08:00
PStatement scopeTypeStatement = mCurrentScope;
2021-08-25 00:20:07 +08:00
while (scopeTypeStatement && !isScopeTypeKind(scopeTypeStatement->kind)) {
scopeTypeStatement = scopeTypeStatement->parentScope.lock();
}
if (
2021-12-03 20:24:49 +08:00
(memberOperator != "::")
2021-08-25 00:20:07 +08:00
&& (
2021-12-08 19:13:47 +08:00
ownerStatement->kind == EvalStatementKind::Variable)
2021-08-25 00:20:07 +08:00
) {
// Get type statement of current (scope) statement
2021-12-08 19:13:47 +08:00
PStatement classTypeStatement = ownerStatement->effectiveTypeStatement;
2021-08-25 00:20:07 +08:00
if (!classTypeStatement)
return;
// It's a iterator
if (ownerStatement
&& ownerStatement->typeStatement
&& STLIterators.contains(ownerStatement->typeStatement->command)
&& (memberOperator == "->"
|| memberOperator == "->*")
) {
PStatement parentScope = ownerStatement->typeStatement->parentScope.lock();
if (STLContainers.contains(parentScope->fullName)) {
QString typeName=mParser->findFirstTemplateParamOf(fileName,ownerStatement->templateParams, parentScope);
// qDebug()<<"typeName"<<typeName<<lastResult->baseStatement->type<<lastResult->baseStatement->command;
classTypeStatement=mParser->findTypeDefinitionOf(fileName, typeName,parentScope);
if (!classTypeStatement)
return;
} else if (STLMaps.contains(parentScope->fullName)) {
QString typeName="std::pair";
classTypeStatement=mParser->findTypeDefinitionOf(fileName, typeName,parentScope);
if (!classTypeStatement)
return;
}
2023-06-20 10:58:14 +08:00
} else if (STLPointers.contains(classTypeStatement->fullName)
2021-12-03 20:24:49 +08:00
&& (memberOperator == "->"
2021-12-08 21:44:40 +08:00
|| memberOperator == "->*")
&& ownerStatement->baseStatement) {
2023-06-20 10:58:14 +08:00
//is a smart pointer
2021-08-25 00:20:07 +08:00
QString typeName= mParser->findFirstTemplateParamOf(
fileName,
2021-12-08 21:44:40 +08:00
ownerStatement->baseStatement->type,
2021-12-08 19:13:47 +08:00
scope);
2021-08-25 00:20:07 +08:00
classTypeStatement = mParser->findTypeDefinitionOf(
fileName,
typeName,
2021-12-08 19:13:47 +08:00
scope);
2021-08-25 00:20:07 +08:00
if (!classTypeStatement)
return;
2023-06-20 10:58:14 +08:00
} else {
//normal member access
if (memberOperator=="." && ownerStatement->pointerLevel !=0)
return;
if (memberOperator=="->" && ownerStatement->pointerLevel!=1)
return;
2021-08-25 00:20:07 +08:00
}
if (!isIncluded(classTypeStatement->fileName) &&
!isIncluded(classTypeStatement->definitionFileName))
return;
2021-12-08 19:13:47 +08:00
if ((classTypeStatement == scopeTypeStatement) || (ownerStatement->effectiveTypeStatement->command == "this")) {
2021-08-25 00:20:07 +08:00
//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;
2021-08-29 00:48:23 +08:00
foreach (const PStatement& childStatement, children) {
if ((childStatement->accessibility==StatementAccessibility::Public)
2021-08-25 00:20:07 +08:00
&& !(
2024-04-03 10:37:29 +08:00
childStatement->kind == StatementKind::Constructor
|| childStatement->kind == StatementKind::Destructor)
2021-08-25 00:20:07 +08:00
&& !mAddedStatements.contains(childStatement->command)) {
addStatement(childStatement,fileName,-1);
}
}
}
//todo friend
2021-12-03 20:24:49 +08:00
} else if ((memberOperator == "::")
2021-12-08 19:13:47 +08:00
&& (ownerStatement->kind == EvalStatementKind::Type)) {
2021-08-25 00:20:07 +08:00
//we can add all child enum definess
2021-12-08 19:13:47 +08:00
PStatement classTypeStatement = ownerStatement->effectiveTypeStatement;
if (!classTypeStatement)
2021-08-25 00:20:07 +08:00
return;
if (!isIncluded(classTypeStatement->fileName) &&
!isIncluded(classTypeStatement->definitionFileName))
return;
2024-04-03 10:37:29 +08:00
if (classTypeStatement->kind == StatementKind::EnumType
|| classTypeStatement->kind == StatementKind::EnumClassType) {
2021-08-25 00:20:07 +08:00
const StatementMap& children =
mParser->statementList().childrenStatements(classTypeStatement);
2021-12-08 19:13:47 +08:00
foreach (const PStatement& child,children) {
addStatement(child,fileName,line);
2021-08-25 00:20:07 +08:00
}
} else {
2021-12-08 19:13:47 +08:00
//class
if (classTypeStatement == scopeTypeStatement) {
//we can use all static members
const StatementMap& children =
mParser->statementList().childrenStatements(classTypeStatement);
foreach (const PStatement& childStatement, children) {
if (
2022-11-16 10:29:20 +08:00
(childStatement->isStatic())
2024-04-03 10:37:29 +08:00
|| (childStatement->kind == StatementKind::Typedef
|| childStatement->kind == StatementKind::Class
|| childStatement->kind == StatementKind::Enum
|| childStatement->kind == StatementKind::EnumClassType
|| childStatement->kind == StatementKind::EnumType
2021-12-08 19:13:47 +08:00
)) {
2021-08-25 00:20:07 +08:00
addStatement(childStatement,fileName,-1);
2021-12-08 19:13:47 +08:00
}
}
} else {
// we can only use public static members
const StatementMap& children =
mParser->statementList().childrenStatements(classTypeStatement);
foreach (const PStatement& childStatement,children) {
if (
2022-11-16 10:29:20 +08:00
(childStatement->isStatic())
2024-04-03 10:37:29 +08:00
|| (childStatement->kind == StatementKind::Typedef
|| childStatement->kind == StatementKind::Class
|| childStatement->kind == StatementKind::Enum
|| childStatement->kind == StatementKind::EnumClassType
|| childStatement->kind == StatementKind::EnumType
2021-12-08 19:13:47 +08:00
)) {
if (childStatement->accessibility == StatementAccessibility::Public)
2021-12-08 19:13:47 +08:00
addStatement(childStatement,fileName,-1);
}
2021-08-25 00:20:07 +08:00
}
}
}
}
}
2021-08-24 15:05:10 +08:00
}
2021-08-25 00:20:07 +08:00
}
void CodeCompletionPopup::getCompletionForFunctionWithoutDefinition(const QString& preWord, QStringList ownerExpression, const QString &memberOperator, const QStringList &memberExpression, const QString &fileName, int line)
{
if(!mParser) {
return;
}
if (!mParser->enabled())
return;
if (memberOperator.isEmpty() && ownerExpression.isEmpty() && memberExpression.isEmpty())
return;
if (!mParser->freeze())
return;
{
auto action = finally([this]{
mParser->unFreeze();
});
if (memberOperator.isEmpty()) {
getCompletionListForComplexKeyword(preWord);
PStatement scopeStatement = mCurrentScope;
//add members of current scope that not added before
2024-04-03 10:37:29 +08:00
while (scopeStatement && scopeStatement->kind!=StatementKind::Namespace
&& scopeStatement->kind!=StatementKind::Class) {
scopeStatement = scopeStatement->parentScope.lock();
}
if (scopeStatement) {
2024-04-03 10:37:29 +08:00
if (scopeStatement->kind == StatementKind::Namespace) {
//namespace;
PStatementList namespaceStatementsList =
mParser->findNamespace(scopeStatement->fullName);
if (namespaceStatementsList) {
foreach (const PStatement& namespaceStatement, *namespaceStatementsList) {
addFunctionWithoutDefinitionChildren(namespaceStatement, fileName, line);
}
}
} else {
//class
addKeyword("operator");
addFunctionWithoutDefinitionChildren(scopeStatement, fileName, line);
}
} else {
//global
addFunctionWithoutDefinitionChildren(scopeStatement, fileName, line);
}
} else {
if (memberOperator != "::")
return;
//the identifier to be completed is a member of variable/class
if (ownerExpression.isEmpty()) {
// start with '::', we only find in global
// add all global members and not added before
addFunctionWithoutDefinitionChildren(nullptr, fileName, line);
return;
}
if (memberExpression.length()==2 && memberExpression.front()!="~")
return;
if (memberExpression.length()>2)
return;
PStatement scope = mCurrentScope;//the scope the expression in
PStatement parentTypeStatement;
PEvalStatement ownerStatement = mParser->evalExpression(fileName,
ownerExpression,
scope);
if(!ownerStatement || !ownerStatement->effectiveTypeStatement) {
return;
}
if (ownerStatement->kind==EvalStatementKind::Namespace) {
//there might be many statements corresponding to one namespace;
PStatementList namespaceStatementsList =
mParser->findNamespace(ownerStatement->baseType);
if (namespaceStatementsList) {
foreach (const PStatement& namespaceStatement, *namespaceStatementsList) {
addFunctionWithoutDefinitionChildren(namespaceStatement, fileName, line);
}
}
return;
2024-04-03 10:37:29 +08:00
} else if (ownerStatement->effectiveTypeStatement->kind == StatementKind::Class) {
addKeyword("operator");
addFunctionWithoutDefinitionChildren(ownerStatement->effectiveTypeStatement, fileName, line);
}
}
}
}
void CodeCompletionPopup::getCompletionListForComplexKeyword(const QString &preWord)
{
mFullCompletionStatementList.clear();
if (preWord == "long") {
addKeyword("long");
addKeyword("double");
addKeyword("int");
} else if (preWord == "short") {
addKeyword("int");
} else if (preWord == "signed") {
addKeyword("long");
addKeyword("short");
addKeyword("int");
addKeyword("char");
} else if (preWord == "unsigned") {
addKeyword("long");
addKeyword("short");
addKeyword("int");
addKeyword("char");
} else if (preWord == "using") {
addKeyword("namespace");
}
}
2022-12-03 11:02:33 +08:00
void CodeCompletionPopup::getCompletionListForNamespaces(const QString &/*preWord*/,
const QString& fileName,
int line)
{
if (!mParser->enabled())
return;
if (!mParser->freeze())
return;
{
auto action = finally([this]{
mParser->unFreeze();
});
QList<QString> namespaceNames = mParser->namespaces();
foreach (const QString& name, namespaceNames) {
PStatementList namespaces = mParser->findNamespace(name);
foreach(const PStatement& statement, *namespaces) {
if (isIncluded(statement->fileName)
|| isIncluded(statement->definitionFileName)) {
addStatement(statement,fileName,line);
continue;
}
}
}
}
}
void CodeCompletionPopup::getCompletionListForTypes(const QString &preWord, const QString &fileName, int line)
{
if (preWord=="typedef") {
addKeyword("const");
addKeyword("struct");
addKeyword("class");
}
if (mShowKeywords) {
//add keywords
foreach (const QString& keyword,CppTypeKeywords) {
addKeyword(keyword);
}
}
if (!mParser->enabled())
return;
if (!mParser->freeze())
return;
{
auto action = finally([this]{
mParser->unFreeze();
});
QList<PStatement> statements = mParser->listTypeStatements(fileName,line);
foreach(const PStatement& statement, statements) {
if (isIncluded(statement->fileName)
|| isIncluded(statement->definitionFileName)) {
addStatement(statement,fileName,line);
}
}
}
}
void CodeCompletionPopup::addKeyword(const QString &keyword)
{
PStatement statement = std::make_shared<Statement>();
statement->command = keyword;
2024-04-03 10:37:29 +08:00
statement->kind = StatementKind::Keyword;
statement->fullName = keyword;
statement->usageCount = 0;
mFullCompletionStatementList.append(statement);
}
2021-08-29 00:48:23 +08:00
bool CodeCompletionPopup::isIncluded(const QString &fileName)
2021-08-25 00:20:07 +08:00
{
return mIncludedFiles.contains(fileName);
}
void CodeCompletionPopup::setHideSymbolsStartWithTwoUnderline(bool newHideSymbolsStartWithTwoUnderline)
{
mHideSymbolsStartWithTwoUnderline = newHideSymbolsStartWithTwoUnderline;
}
void CodeCompletionPopup::setLineHeightFactor(float factor)
{
mDelegate->setLineHeightFactor(factor);
}
bool CodeCompletionPopup::hideSymbolsStartWithTwoUnderline() const
{
return mHideSymbolsStartWithTwoUnderline;
}
bool CodeCompletionPopup::hideSymbolsStartWithUnderline() const
{
return mHideSymbolsStartWithUnderline;
}
void CodeCompletionPopup::setHideSymbolsStartWithUnderline(bool newHideSymbolsStartWithUnderline)
{
mHideSymbolsStartWithUnderline = newHideSymbolsStartWithUnderline;
}
2021-12-04 10:02:07 +08:00
const QString &CodeCompletionPopup::memberOperator() const
{
return mMemberOperator;
}
2021-09-30 20:10:48 +08:00
const QList<PCodeSnippet> &CodeCompletionPopup::codeSnippets() const
{
return mCodeSnippets;
}
void CodeCompletionPopup::setCodeSnippets(const QList<PCodeSnippet> &newCodeSnippets)
{
mCodeSnippets = newCodeSnippets;
}
void CodeCompletionPopup::setColors(const std::shared_ptr<QHash<StatementKind, std::shared_ptr<ColorSchemeItem> > > &newColors)
2021-09-25 23:12:36 +08:00
{
mColors = newColors;
}
2021-12-04 10:02:07 +08:00
const QString &CodeCompletionPopup::memberPhrase() const
{
2021-12-04 10:02:07 +08:00
return mMemberPhrase;
}
2022-11-10 08:05:04 +08:00
const PStatement &CodeCompletionPopup::currentScope() const
2021-08-25 23:53:35 +08:00
{
2022-11-10 08:05:04 +08:00
return mCurrentScope;
2021-08-25 23:53:35 +08:00
}
2022-11-10 08:05:04 +08:00
void CodeCompletionPopup::setCurrentScope(const PStatement &newCurrentStatement)
2021-08-25 23:53:35 +08:00
{
2022-11-10 08:05:04 +08:00
mCurrentScope = newCurrentStatement;
2021-08-25 23:53:35 +08:00
}
const std::shared_ptr<QHash<StatementKind, std::shared_ptr<ColorSchemeItem> > >& CodeCompletionPopup::colors() const
2021-08-28 09:01:40 +08:00
{
return mColors;
}
2021-08-29 00:48:23 +08:00
bool CodeCompletionPopup::sortByScope() const
2021-08-25 23:53:35 +08:00
{
return mSortByScope;
}
2021-08-29 00:48:23 +08:00
void CodeCompletionPopup::setSortByScope(bool newSortByScope)
2021-08-25 23:53:35 +08:00
{
mSortByScope = newSortByScope;
}
2021-08-29 00:48:23 +08:00
bool CodeCompletionPopup::ignoreCase() const
2021-08-25 23:53:35 +08:00
{
return mIgnoreCase;
}
2021-08-29 00:48:23 +08:00
void CodeCompletionPopup::setIgnoreCase(bool newIgnoreCase)
2021-08-25 23:53:35 +08:00
{
mIgnoreCase = newIgnoreCase;
}
2021-09-30 21:25:48 +08:00
bool CodeCompletionPopup::showCodeSnippets() const
2021-08-25 23:53:35 +08:00
{
2021-09-30 20:10:48 +08:00
return mShowCodeSnippets;
2021-08-25 23:53:35 +08:00
}
2021-09-30 21:25:48 +08:00
void CodeCompletionPopup::setShowCodeSnippets(bool newShowCodeIns)
2021-08-25 23:53:35 +08:00
{
2021-09-30 20:10:48 +08:00
mShowCodeSnippets = newShowCodeIns;
2021-08-25 23:53:35 +08:00
}
2021-08-29 00:48:23 +08:00
bool CodeCompletionPopup::showKeywords() const
2021-08-25 23:53:35 +08:00
{
return mShowKeywords;
}
2021-08-29 00:48:23 +08:00
void CodeCompletionPopup::setShowKeywords(bool newShowKeywords)
2021-08-25 23:53:35 +08:00
{
mShowKeywords = newShowKeywords;
}
2021-08-29 00:48:23 +08:00
bool CodeCompletionPopup::recordUsage() const
2021-08-25 23:53:35 +08:00
{
return mRecordUsage;
}
2021-08-29 00:48:23 +08:00
void CodeCompletionPopup::setRecordUsage(bool newRecordUsage)
2021-08-25 23:53:35 +08:00
{
mRecordUsage = newRecordUsage;
}
2021-08-29 00:48:23 +08:00
int CodeCompletionPopup::showCount() const
2021-08-25 23:53:35 +08:00
{
return mShowCount;
}
2021-08-29 00:48:23 +08:00
void CodeCompletionPopup::setShowCount(int newShowCount)
2021-08-25 23:53:35 +08:00
{
mShowCount = newShowCount;
}
2021-08-29 00:48:23 +08:00
const PCppParser &CodeCompletionPopup::parser() const
2021-08-25 23:53:35 +08:00
{
return mParser;
}
2021-08-29 00:48:23 +08:00
void CodeCompletionPopup::setParser(const PCppParser &newParser)
2021-08-25 23:53:35 +08:00
{
mParser = newParser;
}
2021-08-29 00:48:23 +08:00
void CodeCompletionPopup::hideEvent(QHideEvent *event)
2021-08-25 00:20:07 +08:00
{
QMutexLocker locker(&mMutex);
2021-08-25 08:48:33 +08:00
mListView->setKeypressedCallback(nullptr);
2021-08-25 00:20:07 +08:00
mCompletionStatementList.clear();
2022-04-25 10:13:28 +08:00
// foreach (PStatement statement, mFullCompletionStatementList) {
// statement->matchPositions.clear();
// }
2021-08-25 00:20:07 +08:00
mFullCompletionStatementList.clear();
mIncludedFiles.clear();
mUsings.clear();
mAddedStatements.clear();
2022-11-10 08:05:04 +08:00
mCurrentScope = nullptr;
mParser = nullptr;
2021-08-25 00:20:07 +08:00
QWidget::hideEvent(event);
2021-08-24 15:05:10 +08:00
}
2021-08-29 00:48:23 +08:00
bool CodeCompletionPopup::event(QEvent *event)
{
2021-08-27 16:38:55 +08:00
bool result = QWidget::event(event);
if (event->type() == QEvent::FontChange) {
mListView->setFont(font());
mDelegate->setFont(font());
}
2021-08-27 16:38:55 +08:00
return result;
}
2021-08-29 00:48:23 +08:00
CodeCompletionListModel::CodeCompletionListModel(const StatementList *statements, QObject *parent):
2021-08-25 08:48:33 +08:00
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-28 09:01:40 +08:00
switch(role) {
case Qt::DisplayRole: {
2021-08-25 08:48:33 +08:00
PStatement statement = mStatements->at(index.row());
return statement->command;
}
}
2021-08-25 08:48:33 +08:00
return QVariant();
}
PStatement CodeCompletionListModel::statement(const QModelIndex &index) const
{
if (!index.isValid())
return PStatement();
if (index.row()>=mStatements->count())
return PStatement();
return mStatements->at(index.row());
}
QPixmap CodeCompletionListModel::statementIcon(const QModelIndex &index, int size) const
{
if (!index.isValid())
return QPixmap();
if (index.row()>=mStatements->count())
return QPixmap();
PStatement statement = mStatements->at(index.row());
return pIconsManager->getPixmapForStatement(statement, size);
}
2021-08-25 08:48:33 +08:00
void CodeCompletionListModel::notifyUpdated()
{
beginResetModel();
endResetModel();
}
2021-08-28 09:01:40 +08:00
void CodeCompletionListItemDelegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const
{
PStatement statement;
if (mModel && (statement = mModel->statement(index)) ) {
QFont normalFont{font()};
QFont matchedFont{font()};
normalFont.setBold(false);
matchedFont.setBold(true);
painter->save();
painter->setFont(normalFont);
2022-01-27 21:27:51 +08:00
QColor normalColor = mNormalColor;
QColor matchedColor = mMatchedColor;
2022-01-27 21:27:51 +08:00
if (option.state & QStyle::State_Selected) {
painter->fillRect(option.rect, mCurrentSelectionColor);
2022-01-27 21:27:51 +08:00
}
2024-04-07 09:58:30 +08:00
QFontMetrics fm{font()};
2024-04-07 10:22:57 +08:00
int iconSize = fm.height()*0.8;
2024-04-07 09:58:30 +08:00
QPixmap icon = mModel->statementIcon(index, iconSize);
int x=option.rect.left();
if (!icon.isNull()) {
qreal dpr=icon.devicePixelRatioF();
int x1=x+(option.rect.height()-icon.width()/dpr)/2;
int y1=option.rect.top()+(option.rect.height()-icon.height()/dpr)/2;
painter->drawPixmap(x1,y1,icon);
2022-01-27 21:27:51 +08:00
x+=option.rect.height();
}
QString text = statement->command;
int pos=0;
int padding = (option.rect.height()-painter->fontMetrics().height())/2;
int y=option.rect.bottom()-painter->fontMetrics().descent()-padding;
foreach (const PStatementMathPosition& matchPosition, statement->matchPositions) {
if (pos<matchPosition->start) {
QString t = text.mid(pos,matchPosition->start-pos);
2022-01-27 21:27:51 +08:00
painter->setPen(normalColor);
painter->setFont(normalFont);
painter->drawText(x,y,t);
x+=painter->fontMetrics().horizontalAdvance(t);
}
QString t = text.mid(matchPosition->start, matchPosition->end-matchPosition->start);
painter->setPen(matchedColor);
painter->setFont(matchedFont);
painter->drawText(x,y,t);
x+=painter->fontMetrics().horizontalAdvance(t);
pos=matchPosition->end;
}
if (pos<text.length()) {
QString t = text.mid(pos,text.length()-pos);
2022-01-27 21:27:51 +08:00
painter->setPen(normalColor);
painter->setFont(normalFont);
painter->drawText(x,y,t);
x+=painter->fontMetrics().horizontalAdvance(t);
}
painter->restore();
} else {
QStyledItemDelegate::paint(painter, option, index);
}
}
const QColor &CodeCompletionListItemDelegate::normalColor() const
{
return mNormalColor;
}
void CodeCompletionListItemDelegate::setNormalColor(const QColor &newNormalColor)
{
mNormalColor = newNormalColor;
}
const QColor &CodeCompletionListItemDelegate::matchedColor() const
{
return mMatchedColor;
}
void CodeCompletionListItemDelegate::setMatchedColor(const QColor &newMatchedColor)
2021-08-28 09:01:40 +08:00
{
mMatchedColor = newMatchedColor;
2021-08-28 09:01:40 +08:00
}
const QFont &CodeCompletionListItemDelegate::font() const
{
return mFont;
}
void CodeCompletionListItemDelegate::setFont(const QFont &newFont)
{
mFont = newFont;
}
QSize CodeCompletionListItemDelegate::sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const
{
QSize size = QStyledItemDelegate::sizeHint(option, index);
size.setHeight(QFontMetrics(mFont).height()*mLineHeightFactor);
return size;
}
float CodeCompletionListItemDelegate::lineHeightFactor() const
{
return mLineHeightFactor;
}
void CodeCompletionListItemDelegate::setLineHeightFactor(float newLineHeightFactor)
{
mLineHeightFactor = newLineHeightFactor;
}
QColor CodeCompletionListItemDelegate::currentSelectionColor() const
{
return mCurrentSelectionColor;
}
void CodeCompletionListItemDelegate::setCurrentSelectionColor(const QColor &newCurrentSelectionColor)
{
mCurrentSelectionColor = newCurrentSelectionColor;
}
CodeCompletionListItemDelegate::CodeCompletionListItemDelegate(CodeCompletionListModel *model, QWidget *parent) : QStyledItemDelegate(parent),
mModel(model)
2021-08-28 09:01:40 +08:00
{
mNormalColor = qApp->palette().color(QPalette::Text);
mMatchedColor = qApp->palette().color(QPalette::BrightText);
mLineHeightFactor = 1.0;
2021-08-28 09:01:40 +08:00
}