RedPanda-CPP/RedPandaIDE/cpprefacter.cpp

305 lines
11 KiB
C++
Raw Normal View History

#include "cpprefacter.h"
#include "mainwindow.h"
#include "settings.h"
#include "editor.h"
#include "editorlist.h"
#include <QFile>
2021-10-04 11:07:35 +08:00
#include <QMessageBox>
#include <QTextCodec>
#include "HighlighterManager.h"
2021-10-04 11:07:35 +08:00
#include "project.h"
CppRefacter::CppRefacter(QObject *parent) : QObject(parent)
{
}
bool CppRefacter::findOccurence(Editor *editor, const BufferCoord &pos)
{
if (!editor->parser())
return false;
if (!editor->parser()->freeze())
return false;
auto action = finally([&editor]{
editor->parser()->unFreeze();
});
// get full phrase (such as s.name instead of name)
QStringList expression = editor->getExpressionAtPosition(pos);
// Find it's definition
PStatement statement = editor->parser()->findStatementOf(
editor->filename(),
expression,
editor->parser()->findAndScanBlockAt(editor->filename(),pos.Line));
// definition of the symbol not found
if (!statement)
return false;
2021-10-04 11:07:35 +08:00
std::shared_ptr<Project> project = pMainWindow->project();
if (editor->inProject() && project) {
doFindOccurenceInProject(statement,project,editor->parser());
2021-10-04 11:07:35 +08:00
} else {
doFindOccurenceInEditor(statement,editor,editor->parser());
2021-10-04 11:07:35 +08:00
}
pMainWindow->searchResultModel()->notifySearchResultsUpdated();
return true;
}
bool CppRefacter::findOccurence(const QString &statementFullname, SearchFileScope scope)
{
PCppParser parser;
Editor * editor=nullptr;
std::shared_ptr<Project> project;
if (scope == SearchFileScope::currentFile) {
editor = pMainWindow->editorList()->getEditor();
if (!editor)
return false;
parser = editor->parser();
} else if (scope == SearchFileScope::wholeProject) {
project = pMainWindow->project();
if (!project)
return false;
parser = project->cppParser();
}
if (!parser)
return false;
{
parser->freeze();
auto action = finally([&parser]{
parser->unFreeze();
});
PStatement statement = parser->findStatement(statementFullname);
// definition of the symbol not found
if (!statement)
return false;
if (scope == SearchFileScope::wholeProject) {
doFindOccurenceInProject(statement,project,parser);
} else if (scope == SearchFileScope::currentFile) {
doFindOccurenceInEditor(statement, editor,parser);
}
pMainWindow->searchResultModel()->notifySearchResultsUpdated();
return true;
}
}
2021-10-04 11:07:35 +08:00
static QString fullParentName(PStatement statement) {
PStatement parent = statement->parentScope.lock();
if (parent) {
return parent->fullName;
} else {
return "";
}
}
void CppRefacter::renameSymbol(Editor *editor, const BufferCoord &pos, const QString &word, const QString &newWord)
{
if (!editor->parser()->freeze())
return;
auto action = finally([&editor]{
editor->parser()->unFreeze();
});
// get full phrase (such as s.name instead of name)
BufferCoord pBeginPos,pEndPos;
QString phrase = getWordAtPosition(editor,pos,pBeginPos,pEndPos,Editor::WordPurpose::wpInformation);
// Find it's definition
PStatement oldStatement = editor->parser()->findStatementOf(
editor->filename(),
phrase,
pos.Line);
QString oldScope = fullParentName(oldStatement);
// definition of the symbol not found
if (!oldStatement)
return;
// found but not in this file
if (editor->filename() != oldStatement->fileName
|| editor->filename() != oldStatement->definitionFileName) {
QMessageBox::critical(editor,
tr("Rename Symbol Error"),
tr("Can't rename symbols not defined in this file."));
return;
}
2021-10-04 11:07:35 +08:00
QString newPhrase = phrase.mid(0,phrase.length()-word.length()) + newWord;
PStatement newStatement = editor->parser()->findStatementOf(
editor->filename(),
2021-10-04 11:07:35 +08:00
newPhrase,
pos.Line);
if (newStatement && fullParentName(newStatement) == oldScope) {
QMessageBox::critical(editor,
tr("Rename Symbol Error"),
tr("New symbol already exists!"));
return;
}
2021-10-20 18:05:43 +08:00
renameSymbolInFile(editor->filename(),oldStatement,newWord, editor->parser());
}
void CppRefacter::doFindOccurenceInEditor(PStatement statement , Editor *editor, const PCppParser &parser)
{
PSearchResults results = pMainWindow->searchResultModel()->addSearchResults(
statement->command,
statement->fullName,
SearchFileScope::currentFile
);
PSearchResultTreeItem item = findOccurenceInFile(
editor->filename(),
statement,
parser);
if (item && !(item->results.isEmpty())) {
results->results.append(item);
}
}
void CppRefacter::doFindOccurenceInProject(PStatement statement, std::shared_ptr<Project> project, const PCppParser &parser)
{
PSearchResults results = pMainWindow->searchResultModel()->addSearchResults(
statement->command,
statement->fullName,
SearchFileScope::wholeProject
);
foreach (const PProjectUnit& unit, project->units()) {
if (isCfile(unit->fileName()) || isHfile(unit->fileName())) {
PSearchResultTreeItem item = findOccurenceInFile(
unit->fileName(),
statement,
parser);
if (item && !(item->results.isEmpty())) {
results->results.append(item);
}
}
}
}
PSearchResultTreeItem CppRefacter::findOccurenceInFile(
const QString &filename,
const PStatement &statement,
const PCppParser& parser)
{
PSearchResultTreeItem parentItem = std::make_shared<SearchResultTreeItem>();
parentItem->filename = filename;
parentItem->parent = nullptr;
QStringList buffer;
SynEdit editor;
if (pMainWindow->editorList()->getContentFromOpenedEditor(
filename,buffer)){
editor.lines()->setContents(buffer);
} else {
QByteArray encoding;
editor.lines()->loadFromFile(filename,ENCODING_AUTO_DETECT,encoding);
}
editor.setHighlighter(HighlighterManager().getCppHighlighter());
int posY = 0;
while (posY < editor.lines()->count()) {
QString line = editor.lines()->getString(posY);
if (line.isEmpty()) {
posY++;
continue;
}
if (posY == 0) {
editor.highlighter()->resetState();
} else {
editor.highlighter()->setState(
editor.lines()->ranges(posY-1));
}
editor.highlighter()->setLine(line,posY);
while (!editor.highlighter()->eol()) {
int start = editor.highlighter()->getTokenPos() + 1;
QString token = editor.highlighter()->getToken();
if (token == statement->command) {
//same name symbol , test if the same statement;
BufferCoord p,pBeginPos,pEndPos;
p.Line = posY+1;
p.Char = start;
QString phrase = getWordAtPosition(&editor, p, pBeginPos,pEndPos,
Editor::WordPurpose::wpInformation);
PStatement tokenStatement = parser->findStatementOf(
filename,
phrase, p.Line);
if (tokenStatement
&& (tokenStatement->line == statement->line)
&& (tokenStatement->fileName == statement->fileName)) {
PSearchResultTreeItem item = std::make_shared<SearchResultTreeItem>();
item->filename = filename;
item->line = p.Line;
item->start = start;
item->len = phrase.length();
item->parent = parentItem.get();
item->text = line;
item->text.replace('\t',' ');
parentItem->results.append(item);
}
}
editor.highlighter()->next();
}
posY++;
}
return parentItem;
}
2021-10-04 11:07:35 +08:00
2021-10-20 18:05:43 +08:00
void CppRefacter::renameSymbolInFile(const QString &filename, const PStatement &statement, const QString &newWord, const PCppParser &parser)
2021-10-04 11:07:35 +08:00
{
QStringList buffer;
SynEdit editor;
if (pMainWindow->editorList()->getContentFromOpenedEditor(
filename,buffer)){
editor.lines()->setContents(buffer);
} else {
QByteArray encoding;
editor.lines()->loadFromFile(filename,ENCODING_AUTO_DETECT,encoding);
2021-10-04 11:07:35 +08:00
}
QStringList newContents;
editor.setHighlighter(HighlighterManager().getCppHighlighter());
int posY = 0;
while (posY < editor.lines()->count()) {
QString line = editor.lines()->getString(posY);
if (line.isEmpty()) {
posY++;
continue;
}
if (posY == 0) {
editor.highlighter()->resetState();
} else {
editor.highlighter()->setState(
editor.lines()->ranges(posY-1));
}
editor.highlighter()->setLine(line,posY);
QString newLine;
while (!editor.highlighter()->eol()) {
int start = editor.highlighter()->getTokenPos() + 1;
QString token = editor.highlighter()->getToken();
if (token == statement->command) {
//same name symbol , test if the same statement;
BufferCoord p,pBeginPos,pEndPos;
p.Line = posY+1;
p.Char = start;
QString phrase = getWordAtPosition(&editor, p, pBeginPos,pEndPos,
Editor::WordPurpose::wpInformation);
PStatement tokenStatement = parser->findStatementOf(
filename,
phrase, p.Line);
if (tokenStatement
&& (tokenStatement->line == statement->line)
&& (tokenStatement->fileName == statement->fileName)) {
token = newWord;
}
}
newLine += token;
editor.highlighter()->next();
}
newContents.append(newLine);
posY++;
}
Editor * oldEditor = pMainWindow->editorList()->getOpenedEditorByFilename(filename);
if (oldEditor) {
oldEditor->selectAll();
oldEditor->setSelText(newContents.join(oldEditor->lineBreak()));
} else {
QByteArray realEncoding;
QFile file(filename);
editor.lines()->saveToFile(file,ENCODING_AUTO_DETECT,
pSettings->editor().useUTF8ByDefault()? ENCODING_UTF8 : QTextCodec::codecForLocale()->name(),
realEncoding);
2021-10-04 11:07:35 +08:00
}
}