/* * 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/>. */ #include "project.h" #include "editor.h" #include "utils.h" #include "systemconsts.h" #include "editorlist.h" #include <parser/cppparser.h> #include "utils.h" #include "qt_utils/charsetinfo.h" #include "projecttemplate.h" #include "systemconsts.h" #include "iconsmanager.h" #include <QFileSystemWatcher> #include <QDir> #include <QFileDialog> #include <QFileInfo> #include <QMessageBox> #include <QTextCodec> #include <QMessageBox> #include <QDirIterator> #include <QMimeDatabase> #include <QDesktopServices> #include <QJsonObject> #include <QJsonArray> #include <QJsonDocument> #include "customfileiconprovider.h" #include <QMimeData> #include "settings.h" #include "vcs/gitrepository.h" Project::Project(const QString &filename, const QString &name, EditorList* editorList, QFileSystemWatcher* fileSystemWatcher, QObject *parent) : QObject(parent), mName(name), mModified(false), mModel(this), mEditorList(editorList), mFileSystemWatcher(fileSystemWatcher) { mFilename = QFileInfo(filename).absoluteFilePath(); mParser = std::make_shared<CppParser>(); mParser->setOnGetFileStream( std::bind( &EditorList::getContentFromOpenedEditor,mEditorList, std::placeholders::_1, std::placeholders::_2)); mFileSystemWatcher->addPath(directory()); } std::shared_ptr<Project> Project::load(const QString &filename, EditorList *editorList, QFileSystemWatcher *fileSystemWatcher, QObject *parent) { std::shared_ptr<Project> project=std::make_shared<Project>(filename, "", editorList, fileSystemWatcher, parent); project->open(); project->mModified = false; resetCppParser(project->mParser, project->mOptions.compilerSet); return project; } std::shared_ptr<Project> Project::create( const QString &filename, const QString &name, EditorList *editorList, QFileSystemWatcher *fileSystemWatcher, const std::shared_ptr<ProjectTemplate> pTemplate, bool useCpp, QObject *parent) { std::shared_ptr<Project> project=std::make_shared<Project>(filename, name, editorList, fileSystemWatcher, parent); SimpleIni ini; ini.SetValue("Project","filename", toByteArray(extractRelativePath(project->directory(), project->mFilename))); ini.SetValue("Project","name", toByteArray(project->mName)); project->mParser->setEnabled(false); if (!project->assignTemplate(pTemplate,useCpp)) return std::shared_ptr<Project>(); resetCppParser(project->mParser, project->mOptions.compilerSet); project->mModified = true; ini.SaveFile(project->mFilename.toLocal8Bit()); return project; } Project::~Project() { mFileSystemWatcher->removePath(directory()); mEditorList->beginUpdate(); foreach (const PProjectUnit& unit, mUnits) { Editor * editor = unitEditor(unit); if (editor) { editor->setProject(nullptr); if (fileExists(directory())) mEditorList->forceCloseEditor(editor); } } mEditorList->endUpdate(); } QString Project::directory() const { QFileInfo fileInfo(mFilename); return fileInfo.absolutePath(); } QString Project::executable() const { QString exeFileName; if (mOptions.overrideOutput && !mOptions.overridenOutput.isEmpty()) { exeFileName = mOptions.overridenOutput; } else { switch(mOptions.type) { #ifdef ENABLE_SDCC case ProjectType::MicroController: { Settings::PCompilerSet pSet=pSettings->compilerSets().getSet(mOptions.compilerSet); if (pSet) exeFileName = changeFileExt(extractFileName(mFilename),pSet->executableSuffix()); else exeFileName = changeFileExt(extractFileName(mFilename),SDCC_HEX_SUFFIX); } break; #endif case ProjectType::StaticLib: exeFileName = changeFileExt(extractFileName(mFilename),STATIC_LIB_EXT); if (!exeFileName.startsWith("lib")) exeFileName = "lib" + exeFileName; break; case ProjectType::DynamicLib: exeFileName = changeFileExt(extractFileName(mFilename),DYNAMIC_LIB_EXT); if (!exeFileName.startsWith("lib")) exeFileName = "lib" + exeFileName; break; default: exeFileName = changeFileExt(extractFileName(mFilename),DEFAULT_EXECUTABLE_SUFFIX); } } QString exePath; if (!mOptions.exeOutput.isEmpty()) { QDir baseDir(directory()); exePath = baseDir.filePath(mOptions.exeOutput); } else { exePath = directory(); } QDir exeDir(exePath); return exeDir.filePath(exeFileName); } QString Project::makeFileName() { if (mOptions.useCustomMakefile) return mOptions.customMakefile; else return QDir(directory()).filePath(MAKEFILE_NAME); } QString Project::xmakeFileName() { return QDir(directory()).filePath(XMAKEFILE_NAME); } bool Project::unitsModifiedSince(const QDateTime& time) { foreach(const PProjectUnit& unit, mUnits) { QFileInfo info(unit->fileName()); if (info.lastModified()>time) { //qDebug()<<info.lastModified()<<time; return true; } Editor * e=unitEditor(unit); if (e && e->modified()) return true; } return false; } bool Project::modified() const { return mModified; } bool Project::modifiedSince(const QDateTime &time) { if (modified()) return true; QFileInfo info(filename()); return (info.lastModified()>time); } void Project::open() { mModel.beginUpdate(); auto action = finally([this]{ mModel.endUpdate(); }); // QFile fileInfo(mFilename); SimpleIni ini; ini.LoadFile(mFilename.toLocal8Bit()); loadOptions(ini); mRootNode = makeProjectNode(); checkProjectFileForUpdate(ini); int uCount = ini.GetLongValue("Project","UnitCount",0); if (mOptions.modelType==ProjectModelType::FileSystem) { createFileSystemFolderNodes(); } else { createFolderNodes(); } QDir dir(directory()); for (int i=0;i<uCount;i++) { PProjectUnit newUnit = std::make_shared<ProjectUnit>(this); QByteArray groupName = toByteArray(QString("Unit%1").arg(i+1)); newUnit->setFileName( cleanPath(dir.absoluteFilePath( fromByteArray(ini.GetValue(groupName,"FileName",""))))); // if (!QFileInfo(newUnit->fileName()).exists()) { // QMessageBox::critical(nullptr, // tr("File Not Found"), // tr("Project file '%1' can't be found!") // .arg(newUnit->fileName()), // QMessageBox::Ok); // newUnit->setModified(true); // } else { // newUnit->setFileMissing(!QFileInfo(newUnit->fileName()).exists()); newUnit->setFolder(fromByteArray(ini.GetValue(groupName,"Folder",""))); newUnit->setCompile(ini.GetBoolValue(groupName,"Compile", true)); newUnit->setCompileCpp( ini.GetBoolValue(groupName,"CompileCpp",mOptions.isCpp)); newUnit->setLink(ini.GetBoolValue(groupName,"Link", true)); newUnit->setPriority(ini.GetLongValue(groupName,"Priority", 1000)); newUnit->setOverrideBuildCmd(ini.GetBoolValue(groupName,"OverrideBuildCmd", false)); newUnit->setBuildCmd(fromByteArray(ini.GetValue(groupName,"BuildCmd", ""))); newUnit->setEncoding(ini.GetValue(groupName, "FileEncoding",ENCODING_PROJECT)); if (newUnit->encoding()!=ENCODING_UTF16_BOM && newUnit->encoding()!=ENCODING_UTF8_BOM && newUnit->encoding()!=ENCODING_UTF32_BOM && QTextCodec::codecForName(newUnit->encoding())==nullptr) { newUnit->setEncoding(ENCODING_PROJECT); } newUnit->setRealEncoding(ini.GetValue(groupName, "RealEncoding",ENCODING_ASCII)); PProjectModelNode parentNode; if (mOptions.modelType==ProjectModelType::FileSystem) { parentNode = getParentFileSystemFolderNode(newUnit->fileName()); } else { parentNode = getCustomeFolderNodeFromName(newUnit->folder()); } PProjectModelNode node = makeNewFileNode(newUnit, newUnit->priority(), parentNode ); newUnit->setNode(node); mUnits.insert(newUnit->fileName(),newUnit); } } //void Project::setFileName(QString value) //{ // value = QFileInfo(value).absoluteFilePath(); // if (mFilename!=value) { // QFile::rename(mFilename,value); // mFilename = value; // setModified(true); // } //} void Project::setModified(bool value) { if (mModified!=value) { mModified=value; emit modifyChanged(mModified); } } PProjectModelNode Project::makeNewFolderNode( const QString &folderName, PProjectModelNode newParent, ProjectModelNodeType nodeType,int priority) { PProjectModelNode node = std::make_shared<ProjectModelNode>(); if (!newParent) { newParent = mRootNode; } node->parent = newParent; node->text = folderName; if (newParent) { node->level = newParent->level+1; } node->isUnit=false; node->priority = priority; node->folderNodeType = nodeType; QModelIndex parentIndex=mModel.getNodeIndex(newParent.get()); newParent->children.append(node); mModel.insertRow(newParent->children.count()-1,parentIndex); return node; } PProjectModelNode Project::makeNewFileNode(PProjectUnit unit,int priority, PProjectModelNode newParent) { PProjectModelNode node = std::make_shared<ProjectModelNode>(); if (!newParent) { newParent = mRootNode; } node->parent = newParent; node->text = extractFileName(unit->fileName()); if (newParent) { node->level = newParent->level+1; } node->isUnit = true; node->pUnit = unit; node->priority = priority; node->folderNodeType = ProjectModelNodeType::File; newParent->children.append(node); QModelIndex parentIndex=mModel.getNodeIndex(newParent.get()); mModel.insertRow(newParent->children.count()-1,parentIndex); return node; } PProjectModelNode Project::makeProjectNode() { PProjectModelNode node = std::make_shared<ProjectModelNode>(); node->text = mName; node->level = 0; node->isUnit = false; node->folderNodeType = ProjectModelNodeType::Folder; return node; } PProjectUnit Project::newUnit(PProjectModelNode parentNode, const QString& customFileName) { // Select folder to add unit to if (!parentNode) parentNode = mRootNode; // project root node if (parentNode->isUnit) { //it's a file parentNode = mRootNode; } QString s; QDir dir(directory()); // Find unused 'new' filename if (customFileName.isEmpty()) { do { s = cleanPath(dir.absoluteFilePath(QString("untitled%1").arg(getNewFileNumber()))); } while (fileExists(s)); } else { s = cleanPath(dir.absoluteFilePath(customFileName)); } PProjectUnit newUnit = internalAddUnit(s,parentNode); emit unitAdded(newUnit->fileName()); return newUnit; } Editor* Project::openUnit(PProjectUnit& unit, bool forceOpen) { if (!unit->fileName().isEmpty() && fileExists(unit->fileName())) { if (getFileType(unit->fileName())==FileType::Other) { if (forceOpen) QDesktopServices::openUrl(QUrl::fromLocalFile(unit->fileName())); return nullptr; } Editor * editor = mEditorList->getOpenedEditorByFilename(unit->fileName()); if (editor) {//already opened in the editors editor->setProject(this); editor->activate(); return editor; } QByteArray encoding; encoding = unit->encoding(); if (encoding==ENCODING_PROJECT) encoding=options().encoding; editor = mEditorList->newEditor(unit->fileName(), encoding, this, false); if (editor) { //editor->setProject(this); //unit->setEncoding(encoding); loadUnitLayout(editor); editor->activate(); return editor; } } return nullptr; } Editor *Project::openUnit(PProjectUnit &unit, const PProjectEditorLayout &layout) { if (!unit->fileName().isEmpty() && fileExists(unit->fileName())) { if (getFileType(unit->fileName())==FileType::Other) { return nullptr; } Editor * editor = mEditorList->getOpenedEditorByFilename(unit->fileName()); if (editor) {//already opened in the editors editor->setProject(this); editor->activate(); return editor; } QByteArray encoding; encoding = unit->encoding(); if (encoding==ENCODING_PROJECT) encoding=options().encoding; editor = mEditorList->newEditor(unit->fileName(), encoding, this, false); if (editor) { //editor->setInProject(true); editor->setCaretY(layout->caretY); editor->setCaretX(layout->caretX); editor->setTopLine(layout->topLine); editor->setLeftPos(layout->left); editor->activate(); return editor; } } return nullptr; } Editor *Project::unitEditor(const PProjectUnit &unit) const { if (!unit) return nullptr; return mEditorList->getOpenedEditorByFilename(unit->fileName()); } Editor *Project::unitEditor(const ProjectUnit *unit) const { if (!unit) return nullptr; return mEditorList->getOpenedEditorByFilename(unit->fileName()); } QList<PProjectUnit> Project::unitList() { QList<PProjectUnit> units; foreach(PProjectUnit unit, mUnits) { units.append(unit); } return units; } QStringList Project::unitFiles() { QStringList units; foreach(PProjectUnit unit, mUnits) { units.append(unit->fileName()); } return units; } void Project::rebuildNodes() { mModel.beginUpdate(); // Delete everything mRootNode->children.clear(); mCustomFolderNodes.clear(); mSpecialNodes.clear(); mFileSystemFolderNodes.clear(); // Recreate everything switch(mOptions.modelType) { case ProjectModelType::Custom: createFolderNodes(); foreach (PProjectUnit pUnit, mUnits) { QFileInfo fileInfo(pUnit->fileName()); pUnit->setNode( makeNewFileNode( pUnit, pUnit->priority(), getCustomeFolderNodeFromName(pUnit->folder()) ) ); } break; case ProjectModelType::FileSystem: createFileSystemFolderNodes(); foreach (PProjectUnit pUnit, mUnits) { QFileInfo fileInfo(pUnit->fileName()); pUnit->setNode( makeNewFileNode( pUnit, pUnit->priority(), getParentFileSystemFolderNode( pUnit->fileName()) ) ); } break; } mModel.endUpdate(); } bool Project::removeUnit(PProjectUnit& unit, bool doClose , bool removeFile) { bool result=internalRemoveUnit(unit,doClose,removeFile); if (result) { emit unitRemoved(unit->fileName()); } return result; } bool Project::internalRemoveUnit(PProjectUnit& unit, bool doClose , bool removeFile) { if (!unit) return false; // qDebug()<<unit->fileName(); // qDebug()<<(qint64)unit->editor(); // Attempt to close it if (doClose) { Editor* editor = unitEditor(unit); if (editor) { editor->setProject(nullptr); mEditorList->closeEditor(editor); } } if (removeFile) { #if QT_VERSION >= QT_VERSION_CHECK(5,15,2) if (!QFile::moveToTrash(unit->fileName())) QFile::remove(unit->fileName()); #else QFile::remove(unit->fileName()); #endif } //if not fUnits.GetItem(index).fNew then PProjectModelNode node = unit->node(); PProjectModelNode parentNode = node->parent.lock(); if (!parentNode) { mUnits.remove(unit->fileName()); return true; } int row = parentNode->children.indexOf(unit->node()); if (row<0) { mUnits.remove(unit->fileName()); return true; } QModelIndex parentIndex = mModel.getNodeIndex(parentNode.get()); mModel.removeRow(row,parentIndex); mUnits.remove(unit->fileName()); //remove empty parent node PProjectModelNode currentNode = parentNode; while (currentNode && currentNode->folderNodeType == ProjectModelNodeType::Folder && currentNode->children.isEmpty()) { parentNode = currentNode->parent.lock(); if (!parentNode) break; row = parentNode->children.indexOf(currentNode); if (row<0) break; parentIndex = mModel.getNodeIndex(parentNode.get()); mModel.removeRow(row,parentIndex); currentNode = parentNode; } setModified(true); return true; } bool Project::removeFolder(PProjectModelNode node) { mModel.beginUpdate(); auto action = finally([this]{ mModel.endUpdate(); }); // Sanity check if (!node) return false; // Check if this is actually a folder if (node->isUnit || node->level<1) return false; // Let this function call itself removeFolderRecurse(node); // Update list of folders (sets modified) updateFolders(); return true; } void Project::resetParserProjectFiles() { mParser->clearProjectFiles(); mParser->clearProjectIncludePaths(); foreach (const PProjectUnit& unit, mUnits) { if (isCFile(unit->fileName()) || isHFile(unit->fileName())) mParser->addProjectFile(unit->fileName(),true); } foreach (const QString& s, mOptions.includeDirs) { mParser->addProjectIncludePath(s); } } void Project::saveAll() { if (!saveUnits()) return; saveOptions(); // update other data, and save to disk saveLayout(); // save current opened files, and which is "active". // We have saved everything to disk, so mark unmodified setModified(false); } void Project::saveLayout() { if (!fileExists(directory())) return; QHash<QString, PProjectEditorLayout> oldLayouts = loadLayout(); QHash<QString,int> editorOrderSet; // Write list of open project files int order=0; for (int i=0;i<mEditorList->pageCount();i++) { Editor* e=(*mEditorList)[i]; if (e && e->inProject() && !editorOrderSet.contains(e->filename())) { editorOrderSet.insert(e->filename(),order); order++; } } // layIni.SetValue("Editors","Order",sl.join(",").toUtf8()); Editor *e, *e2; // Remember what files were visible mEditorList->getVisibleEditors(e, e2); QJsonArray jsonLayouts; // save editor info foreach (const PProjectUnit& unit,mUnits) { Editor* editor = unitEditor(unit); if (editor) { QJsonObject jsonLayout; jsonLayout["filename"]=unit->fileName(); jsonLayout["caretX"]=editor->caretX(); jsonLayout["caretY"]=editor->caretY(); jsonLayout["topLine"]=editor->topLine(); jsonLayout["left"]=editor->leftPos(); jsonLayout["isOpen"]=true; jsonLayout["focused"]=(editor==e); int order=editorOrderSet.value(editor->filename(),-1); if (order>=0) { jsonLayout["order"]=order; } jsonLayouts.append(jsonLayout); } else { PProjectEditorLayout oldLayout = oldLayouts.value(unit->fileName(),PProjectEditorLayout()); if (oldLayout) { QJsonObject jsonLayout; jsonLayout["filename"]=unit->fileName(); jsonLayout["caretX"]=oldLayout->caretX; jsonLayout["caretY"]=oldLayout->caretY; jsonLayout["topLine"]=oldLayout->topLine; jsonLayout["left"]=oldLayout->left; jsonLayout["isOpen"]=false; jsonLayout["focused"]=false; jsonLayouts.append(jsonLayout); } } } QString jsonFilename = changeFileExt(filename(), "layout"); QFile file(jsonFilename); if (file.open(QFile::WriteOnly|QFile::Truncate)) { QJsonDocument doc(jsonLayouts); file.write(doc.toJson(QJsonDocument::Indented)); file.close(); } } void Project::renameUnit(PProjectUnit& unit, const QString &newFileName) { if (!unit) return; if (newFileName.compare(unit->fileName(),PATH_SENSITIVITY)==0) return; if (mParser) { mParser->removeProjectFile(unit->fileName()); mParser->addProjectFile(newFileName,true); } Editor * editor=unitEditor(unit); if (editor) { //prevent recurse editor->saveAs(newFileName,true); } else { if (mParser) mParser->invalidateFile(unit->fileName()); copyFile(unit->fileName(),newFileName,true); if (mParser) mParser->parseFile(newFileName,true); } internalRemoveUnit(unit,false,true); PProjectModelNode parentNode = unit->node()->parent.lock(); internalAddUnit(newFileName,parentNode); setModified(true); emit unitRenamed(unit->fileName(),newFileName); emit nodeRenamed(); } bool Project::saveUnits() { if (!fileExists(directory())) return false; int count = 0; SimpleIni ini; SI_Error error = ini.LoadFile(mFilename.toLocal8Bit()); if (error != SI_Error::SI_OK) return false; int i=0; foreach (const PProjectUnit& unit, mUnits) { i++; QByteArray groupName = toByteArray(QString("Unit%1").arg(i)); // if (!unit->FileMissing()) { // bool rd_only = false; // if (unit->modified() && fileExists(unit->fileName()) // && isReadOnly(unit->fileName())) { // // file is read-only // QMessageBox::critical(nullptr, // tr("Can't save file"), // tr("Can't save file '%1'").arg(unit->fileName()), // QMessageBox::Ok // ); // rd_only = true; // } // if (!rd_only) { // if (!unit->save() && unit->isNew()) // return false; // } // } // saved new file or an existing file add to project file ini.SetValue( groupName, "FileName", toByteArray( extractRelativePath( directory(), unit->fileName()))); count++; switch(getFileType(unit->fileName())) { case FileType::CHeader: case FileType::CSource: case FileType::CppHeader: case FileType::CppSource: ini.SetLongValue(groupName,"CompileCpp", unit->compileCpp()); break; case FileType::WindowsResourceSource: unit->setFolder("Resources"); default: break; } ini.SetValue(groupName,"Folder", toByteArray(unit->folder())); ini.SetLongValue(groupName,"Compile", unit->compile()); ini.SetLongValue(groupName,"Link", unit->link()); ini.SetLongValue(groupName,"Priority", unit->priority()); ini.SetLongValue(groupName,"OverrideBuildCmd", unit->overrideBuildCmd()); ini.SetValue(groupName,"BuildCmd", toByteArray(unit->buildCmd())); //ini.SetLongValue(groupName,"DetectEncoding", unit->encoding()==ENCODING_AUTO_DETECT); ini.Delete(groupName,"DetectEncoding"); ini.SetValue(groupName,"FileEncoding", unit->encoding()); ini.SetValue(groupName,"RealEncoding",unit->realEncoding()); } ini.SetLongValue("Project","UnitCount",count); ini.SaveFile(mFilename.toLocal8Bit()); return true; } PProjectUnit Project::findUnit(const QString &filename) { return mUnits.value(filename,PProjectUnit()); } PProjectUnit Project::findUnit(const Editor *editor) { if (!editor) return PProjectUnit(); return findUnit(editor->filename()); } void Project::associateEditor(Editor *editor) { PProjectUnit unit = findUnit(editor); associateEditorToUnit(editor,unit); } void Project::associateEditorToUnit(Editor *editor, PProjectUnit unit) { if (!unit) { if (editor) editor->setProject(nullptr); return; } if (editor) { Editor * e= unitEditor(unit); if (e) { if (editor==e) return; e->setProject(nullptr); e->close(); } editor->setProject(this); // if (editor->encodingOption()==ENCODING_AUTO_DETECT) { // if (editor->fileEncoding()==ENCODING_ASCII) { // editor->setEncodingOption(mOptions.encoding); // } else { // editor->setEncodingOption(editor->fileEncoding()); // } // } if (unit->encoding()==ENCODING_PROJECT) { if (editor->encodingOption()!=mOptions.encoding) unit->setEncoding(editor->encodingOption()); } else if (editor->encodingOption()!=unit->encoding()) { unit->setEncoding(editor->encodingOption()); } unit->setRealEncoding(editor->fileEncoding()); } } //bool Project::setCompileOption(const QString &key, int valIndex) //{ // Settings::PCompilerSet pSet = pSettings->compilerSets().getSet(mOptions.compilerSet); // if (!pSet) // return false; // PCompilerOption op = CompilerInfoManager::getCompilerOption( // pSet->compilerType(), key); // if (!op) // return false; // if (op->choices.isEmpty()) { // if (valIndex>0) // mOptions.compilerOptions.insert(key,COMPILER_OPTION_ON); // else // mOptions.compilerOptions.insert(key,""); // } else { // if (valIndex>0 && valIndex <= op->choices.length()) { // mOptions.compilerOptions.insert(key,op->choices[valIndex-1].second); // } else { // mOptions.compilerOptions.insert(key,""); // } // } // return true; //} bool Project::setCompileOption(const QString &key, const QString &value) { Settings::PCompilerSet pSet = pSettings->compilerSets().getSet(mOptions.compilerSet); if (!pSet) return false; PCompilerOption op = CompilerInfoManager::getCompilerOption( pSet->compilerType(), key); if (!op) return false; mOptions.compilerOptions.insert(key,value); return true; } QString Project::getCompileOption(const QString &key) const { return mOptions.compilerOptions.value(key,""); } void Project::updateFolders() { mFolders.clear(); updateFolderNode(mRootNode); foreach (PProjectUnit unit, mUnits) { unit->setFolder( getNodePath( unit->node()->parent.lock() ) ); } setModified(true); } PProjectModelNode Project::pointerToNode(ProjectModelNode *p, PProjectModelNode parent) { if (!p) return PProjectModelNode(); if (!parent) { parent = mRootNode; } if (p==mRootNode.get()) return mRootNode; foreach (const PProjectModelNode& node , parent->children) { if (node.get()==p) return node; PProjectModelNode result = pointerToNode(p,node); if (result) return result; } return PProjectModelNode(); } void Project::setCompilerSet(int compilerSetIndex) { if (mOptions.compilerSet != compilerSetIndex) { mOptions.compilerSet = compilerSetIndex; updateCompilerSetting(); setModified(true); } } bool Project::assignTemplate(const std::shared_ptr<ProjectTemplate> aTemplate, bool useCpp) { if (!aTemplate) { return false; } mModel.beginUpdate(); mRootNode = makeProjectNode(); rebuildNodes(); mOptions = aTemplate->options(); mOptions.compilerSet = pSettings->compilerSets().defaultIndex(); mOptions.isCpp = useCpp; updateCompilerSetting(); mOptions.icon = aTemplate->icon(); QTextCodec* codec=QTextCodec::codecForName(mOptions.encoding); if (!codec) mOptions.encoding=ENCODING_SYSTEM_DEFAULT; codec=QTextCodec::codecForName(mOptions.execEncoding); if (!codec) mOptions.execEncoding=ENCODING_SYSTEM_DEFAULT; // Copy icon to project directory if (!mOptions.icon.isEmpty()) { QString originIcon = cleanPath(QFileInfo(aTemplate->fileName()).absoluteDir().absoluteFilePath(mOptions.icon)); if (fileExists(originIcon)) { QString destIcon = cleanPath(QFileInfo(mFilename).absoluteDir().absoluteFilePath("app.ico")); QFile::copy(originIcon,destIcon); mOptions.icon = destIcon; } else { mOptions.icon = ""; } } // Add list of files if (aTemplate->version() > 0) { QDir dir(aTemplate->folder()); for (int i=0;i<aTemplate->unitCount();i++) { // Pick file contents PTemplateUnit templateUnit = aTemplate->unit(i); if (!templateUnit) continue; if (!templateUnit->Source.isEmpty()) { QString target = templateUnit->Source; PProjectUnit unit; if (!templateUnit->Target.isEmpty()) target = templateUnit->Target; unit = newUnit(mRootNode, target); if (templateUnit->overwrite || !fileExists(unit->fileName()) ) { QFile::copy( cleanPath(dir.absoluteFilePath(templateUnit->Source)), includeTrailingPathDelimiter(this->directory())+target); } FileType fileType=getFileType(unit->fileName()); if ( fileType==FileType::GAS || isCFile(unit->fileName()) || isHFile(unit->fileName())) { Editor * editor = mEditorList->newEditor( unit->fileName(), unit->encoding()==ENCODING_PROJECT?options().encoding:unit->encoding(), this, false); editor->activate(); } } else { QString s; PProjectUnit unit; if (mOptions.isCpp) { s = templateUnit->CppText; unit = newUnit(mRootNode, templateUnit->CppName); } else { s = templateUnit->CText; unit = newUnit(mRootNode,templateUnit->CName); } Editor * editor = mEditorList->newEditor( unit->fileName(), unit->encoding()==ENCODING_PROJECT?options().encoding:unit->encoding(), this, true); if (templateUnit->overwrite || !fileExists(unit->fileName()) ) { QString s2 = cleanPath(dir.absoluteFilePath(s)); if (fileExists(s2) && !s.isEmpty()) { try { editor->loadFile(s2); } catch(FileError& e) { QMessageBox::critical(nullptr, tr("Error Load File"), e.reason()); } } else { s.replace("#13#10","\r\n"); editor->insertString(s,false); } editor->save(true,false); } editor->activate(); } } } mModel.endUpdate(); return true; } bool Project::saveAsTemplate(const QString &templateFolder, const QString& name, const QString& description, const QString& category) { QDir dir(templateFolder); if (!dir.mkpath(templateFolder)) { QMessageBox::critical(nullptr, tr("Error"), tr("Can't create folder %1 ").arg(templateFolder), QMessageBox::Ok); return false; } QString fileName = cleanPath(dir.absoluteFilePath(TEMPLATE_INFO_FILE)); PSimpleIni ini = std::make_shared<SimpleIni>(); ini->SetLongValue("Template","Ver",3); // template info ini->SetValue("Template", "Name", name.toUtf8()); ini->SetValue("Template", "Category", category.toUtf8()); ini->SetValue("Template", "Description", description.toUtf8()); if (fileExists(mOptions.icon)) { QString iconName = extractFileName(mOptions.icon); copyFile(mOptions.icon, cleanPath(dir.absoluteFilePath(iconName)),true); if (dir.exists(iconName)) ini->SetValue("Template", "Icon", iconName.toUtf8()); } ini->SetLongValue("Project", "Type", static_cast<int>(mOptions.type)); if (!mOptions.includeDirs.isEmpty()) ini->SetValue("Project", "Includes", relativePaths(mOptions.includeDirs).join(";").toUtf8()); if (!mOptions.resourceIncludes.isEmpty()) ini->SetValue("Project", "ResourceIncludes", relativePaths(mOptions.resourceIncludes).join(";").toUtf8()); if (!mOptions.makeIncludes.isEmpty()) ini->SetValue("Project", "MakeIncludes", relativePaths(mOptions.makeIncludes).join(";").toUtf8()); if (!mOptions.binDirs.isEmpty()) ini->SetValue("Project", "Bins", relativePaths(mOptions.binDirs).join(";").toUtf8()); if (!mOptions.libDirs.isEmpty()) ini->SetValue("Project", "Libs", relativePaths(mOptions.libDirs).join(";").toUtf8()); if (!mOptions.compilerCmd.isEmpty()) ini->SetValue("Project", "Compiler", textToLines(mOptions.compilerCmd).join(" ").toUtf8()); if (!mOptions.cppCompilerCmd.isEmpty()) ini->SetValue("Project", "CppCompiler", textToLines(mOptions.cppCompilerCmd).join(" ").toUtf8()); if (!mOptions.linkerCmd.isEmpty()) ini->SetValue("Project", "Linker",textToLines(mOptions.linkerCmd).join(" ").toUtf8()); if (!mOptions.resourceCmd.isEmpty()) ini->SetValue("Project", "ResourceCommand",textToLines(mOptions.resourceCmd).join(" ").toUtf8()); ini->SetBoolValue("Project", "IsCpp", mOptions.isCpp); if (mOptions.includeVersionInfo) ini->SetBoolValue("Project", "IncludeVersionInfo", true); if (mOptions.supportXPThemes) ini->SetBoolValue("Project", "SupportXPThemes", true); if (!mOptions.exeOutput.isEmpty()) ini->SetValue("Project", "ExeOutput", extractRelativePath(directory(),mOptions.exeOutput).toUtf8()); if (!mOptions.objectOutput.isEmpty()) ini->SetValue("Project", "ObjectOutput", extractRelativePath(directory(),mOptions.objectOutput).toUtf8()); if (!mOptions.logOutput.isEmpty()) ini->SetValue("Project", "LogOutput", extractRelativePath(directory(),mOptions.logOutput).toUtf8()); if (mOptions.execEncoding!=ENCODING_SYSTEM_DEFAULT) ini->SetValue("Project","ExecEncoding", mOptions.execEncoding); // if (!mOptions.staticLink) // ini->SetBoolValue("Project", "StaticLink",false); if (!mOptions.addCharset) ini->SetBoolValue("Project", "AddCharset",false); if (mOptions.encoding!=ENCODING_AUTO_DETECT) ini->SetValue("Project","Encoding",mOptions.encoding); if (mOptions.modelType!=ProjectModelType::FileSystem) ini->SetLongValue("Project", "ModelType", (int)mOptions.modelType); ini->SetLongValue("Project","ClassBrowserType", (int)mOptions.classBrowserType); int i=0; foreach (const PProjectUnit &unit, mUnits) { QString unitName = extractFileName(unit->fileName()); QByteArray section = toByteArray(QString("Unit%1").arg(i)); if (!copyFile(unit->fileName(), cleanPath(dir.absoluteFilePath(unitName)),true)) { QMessageBox::warning(nullptr, tr("Warning"), tr("Can't save file %1").arg(cleanPath(dir.absoluteFilePath(unitName))), QMessageBox::Ok); } switch(getFileType(unit->fileName())) { case FileType::CSource: ini->SetValue(section,"C", unitName.toUtf8()); ini->SetValue(section,"CName", unitName.toUtf8()); break; case FileType::CppSource: ini->SetValue(section,"Cpp", unitName.toUtf8()); ini->SetValue(section,"CppName", unitName.toUtf8()); break; case FileType::CHeader: case FileType::CppHeader: ini->SetValue(section,"C", unitName.toUtf8()); ini->SetValue(section,"CName", unitName.toUtf8()); ini->SetValue(section,"Cpp", unitName.toUtf8()); ini->SetValue(section,"CppName", unitName.toUtf8()); break; default: ini->SetValue(section,"Source", unitName.toUtf8()); ini->SetValue(section,"Target", unitName.toUtf8()); } i++; } ini->SetLongValue("Project","UnitCount",mUnits.count()); if (ini->SaveFile(fileName.toLocal8Bit())!=SI_OK) { QMessageBox::critical(nullptr, tr("Error"), tr("Can't save file %1").arg(fileName), QMessageBox::Ok); return false; } return true; } void Project::setEncoding(const QByteArray &encoding) { if (encoding!=mOptions.encoding) { mOptions.encoding=encoding; foreach (const PProjectUnit& unit,mUnits) { if (unit->encoding()!=ENCODING_PROJECT) continue; Editor * e=unitEditor(unit); if (e) { e->setEncodingOption(ENCODING_PROJECT); unit->setEncoding(ENCODING_PROJECT); } } } } void Project::saveOptions() { if (!fileExists(directory())) return; SimpleIni ini; ini.LoadFile(mFilename.toLocal8Bit()); ini.SetValue("Project","FileName", toByteArray(extractRelativePath(directory(), mFilename))); ini.SetValue("Project","Name", toByteArray(mName)); ini.SetLongValue("Project","Type", static_cast<int>(mOptions.type)); ini.SetLongValue("Project","Ver", 3); // Is 3 as of Red Panda C++.0 ini.SetValue("Project","Includes", toByteArray(relativePaths(mOptions.includeDirs).join(";"))); ini.SetValue("Project","Libs", toByteArray(relativePaths(mOptions.libDirs).join(";"))); ini.SetValue("Project","Bins", toByteArray(relativePaths(mOptions.binDirs).join(";"))); ini.SetValue("Project","ResourceIncludes", toByteArray(relativePaths(mOptions.resourceIncludes).join(";"))); ini.SetValue("Project","MakeIncludes", toByteArray(relativePaths(mOptions.makeIncludes).join(";"))); ini.SetValue("Project","PrivateResource", toByteArray(mOptions.privateResource)); ini.SetValue("Project","Compiler", toByteArray(textToLines(mOptions.compilerCmd).join(" "))); ini.SetValue("Project","CppCompiler", toByteArray(textToLines(mOptions.cppCompilerCmd).join(" "))); ini.SetValue("Project","Linker", toByteArray(textToLines(mOptions.linkerCmd).join(" "))); ini.SetValue("Project", "ResourceCommand", toByteArray(textToLines(mOptions.resourceCmd).join(" "))); ini.SetLongValue("Project","IsCpp", mOptions.isCpp); ini.SetValue("Project","Icon", toByteArray(extractRelativePath(directory(), mOptions.icon))); ini.SetValue("Project","ExeOutput", toByteArray(extractRelativePath(directory(),mOptions.exeOutput))); ini.SetValue("Project","ObjectOutput", toByteArray(extractRelativePath(directory(),mOptions.objectOutput))); ini.SetValue("Project","LogOutput", toByteArray(extractRelativePath(directory(),mOptions.logOutput))); ini.SetLongValue("Project","LogOutputEnabled", mOptions.logOutputEnabled); ini.SetLongValue("Project","OverrideOutput", mOptions.overrideOutput); ini.SetValue("Project","OverrideOutputName", toByteArray(mOptions.overridenOutput)); ini.SetValue("Project","HostApplication", toByteArray(extractRelativePath(directory(), mOptions.hostApplication))); ini.SetLongValue("Project","UseCustomMakefile", mOptions.useCustomMakefile); ini.SetValue("Project","CustomMakefile", toByteArray(extractRelativePath(directory(),mOptions.customMakefile))); ini.SetLongValue("Project","UsePrecompiledHeader", mOptions.usePrecompiledHeader); ini.SetValue("Project","PrecompiledHeader", toByteArray(extractRelativePath(directory(), mOptions.precompiledHeader))); ini.SetValue("Project","CommandLine", toByteArray(mOptions.cmdLineArgs)); ini.SetValue("Project","Folders", toByteArray(mFolders.join(";"))); ini.SetLongValue("Project","IncludeVersionInfo", mOptions.includeVersionInfo); ini.SetLongValue("Project","SupportXPThemes", mOptions.supportXPThemes); ini.SetLongValue("Project","CompilerSet", mOptions.compilerSet); ini.Delete("Project","CompilerSettings"); // remove old compiler settings ini.Delete("CompilerSettings",nullptr); // remove old compiler settings foreach (const QString& key, mOptions.compilerOptions.keys()) { ini.SetValue("CompilerSettings",toByteArray(key),toByteArray(mOptions.compilerOptions.value(key))); } ini.SetLongValue("Project","StaticLink", mOptions.staticLink); ini.SetLongValue("Project","AddCharset", mOptions.addCharset); ini.SetValue("Project","ExecEncoding", mOptions.execEncoding); ini.SetValue("Project","Encoding",mOptions.encoding); ini.SetLongValue("Project","ModelType", (int)mOptions.modelType); ini.SetLongValue("Project","ClassBrowserType", (int)mOptions.classBrowserType); ini.SetBoolValue("Project","AllowParallelBuilding",mOptions.allowParallelBuilding); ini.SetLongValue("Project","ParellelBuildingJobs",mOptions.parellelBuildingJobs); //for Red Panda Dev C++ 6 compatibility ini.SetLongValue("Project","UseUTF8",mOptions.encoding == ENCODING_UTF8); ini.SetLongValue("VersionInfo","Major", mOptions.versionInfo.major); ini.SetLongValue("VersionInfo","Minor", mOptions.versionInfo.minor); ini.SetLongValue("VersionInfo","Release", mOptions.versionInfo.release); ini.SetLongValue("VersionInfo","Build", mOptions.versionInfo.build); ini.SetLongValue("VersionInfo","LanguageID", mOptions.versionInfo.languageID); ini.SetLongValue("VersionInfo","CharsetID", mOptions.versionInfo.charsetID); ini.SetValue("VersionInfo","CompanyName", toByteArray(mOptions.versionInfo.companyName)); ini.SetValue("VersionInfo","FileVersion", toByteArray(mOptions.versionInfo.fileVersion)); ini.SetValue("VersionInfo","FileDescription", toByteArray(mOptions.versionInfo.fileDescription)); ini.SetValue("VersionInfo","InternalName", toByteArray(mOptions.versionInfo.internalName)); ini.SetValue("VersionInfo","LegalCopyright", toByteArray(mOptions.versionInfo.legalCopyright)); ini.SetValue("VersionInfo","LegalTrademarks", toByteArray(mOptions.versionInfo.legalTrademarks)); ini.SetValue("VersionInfo","OriginalFilename", toByteArray(mOptions.versionInfo.originalFilename)); ini.SetValue("VersionInfo","ProductName", toByteArray(mOptions.versionInfo.productName)); ini.SetValue("VersionInfo","ProductVersion", toByteArray(mOptions.versionInfo.productVersion)); ini.SetLongValue("VersionInfo","AutoIncBuildNr", mOptions.versionInfo.autoIncBuildNr); ini.SetLongValue("VersionInfo","SyncProduct", mOptions.versionInfo.syncProduct); //delete outdated dev4 project options ini.Delete("Project","NoConsole"); ini.Delete("Project","IsDLL"); ini.Delete("Project","ResFiles"); ini.Delete("Project","IncludeDirs"); ini.Delete("Project","CompilerOptions"); ini.Delete("Project","Use_GPP"); ini.SaveFile(mFilename.toLocal8Bit()); } PProjectModelNode Project::addFolder(PProjectModelNode parentFolder,const QString &s) { QString fullPath; QString path = getNodePath(parentFolder); if (path.isEmpty()) { fullPath = s; } else { fullPath = path + '/' +s; } if (mFolders.indexOf(fullPath)<0) { mModel.beginUpdate(); auto action = finally([this]{ mModel.endUpdate(); }); mFolders.append(fullPath); PProjectModelNode node = makeNewFolderNode(s,parentFolder); setModified(true); return node; } return PProjectModelNode(); } PProjectUnit Project::addUnit(const QString &inFileName, PProjectModelNode parentNode) { PProjectUnit newUnit=internalAddUnit(inFileName, parentNode); if (newUnit) { emit unitAdded(newUnit->fileName()); } return newUnit; } PProjectUnit Project::internalAddUnit(const QString &inFileName, PProjectModelNode parentNode) { // Don't add if it already exists if (fileAlreadyExists(inFileName)) { QMessageBox::critical(nullptr, tr("File Exists"), tr("File '%1' is already in the project").arg(inFileName), QMessageBox::Ok); return PProjectUnit(); } if (mOptions.modelType == ProjectModelType::FileSystem) { // in file system mode, parentNode is determined by file's path parentNode = getParentFileSystemFolderNode(inFileName); } PProjectUnit newUnit = std::make_shared<ProjectUnit>(this); // Set all properties newUnit->setFileName(QDir(directory()).filePath(inFileName)); Editor * e= unitEditor(newUnit); if (e) { associateEditorToUnit(e,newUnit); // newUnit->setEncoding(e->encodingOption()); // newUnit->setRealEncoding(e->fileEncoding()); // e->setProject(this); } else { newUnit->setEncoding(ENCODING_PROJECT); } // Determine compilation flags switch(getFileType(inFileName)) { case FileType::GAS: newUnit->setCompile(true); newUnit->setCompileCpp(false); newUnit->setLink(true); break; case FileType::CSource: newUnit->setCompile(true); newUnit->setCompileCpp(false); newUnit->setLink(true); break; case FileType::CppSource: newUnit->setCompile(true); newUnit->setCompileCpp(true); newUnit->setLink(true); break; case FileType::WindowsResourceSource: newUnit->setCompile(true); newUnit->setCompileCpp(mOptions.isCpp); newUnit->setLink(false); break; default: newUnit->setCompile(false); newUnit->setCompileCpp(false); newUnit->setLink(false); } if (mOptions.modelType == ProjectModelType::FileSystem) newUnit->setFolder(getNodePath(parentNode)); newUnit->setPriority(1000); newUnit->setOverrideBuildCmd(false); newUnit->setBuildCmd(""); PProjectModelNode node = makeNewFileNode(newUnit, newUnit->priority(), parentNode); newUnit->setNode(node); mUnits.insert(newUnit->fileName(),newUnit); setModified(true); return newUnit; } QString Project::folder() { return extractFileDir(filename()); } void Project::buildPrivateResource() { int comp = 0; foreach (const PProjectUnit& unit,mUnits) { if ( (getFileType(unit->fileName()) == FileType::WindowsResourceSource) && unit->compile() ) comp++; } // if project has no other resources included // and does not have an icon // and does not include the XP style manifest // and does not include version info // then do not create a private resource file if ((comp == 0) && (! mOptions.supportXPThemes) && (! mOptions.includeVersionInfo) && (mOptions.icon == "")) { mOptions.privateResource=""; return; } // change private resource from <project_filename>.res // to <project_filename>_private.res // // in many cases (like in importing a MSVC project) // the project's resource file has already the // <project_filename>.res filename. QString rcFile; if (!mOptions.privateResource.isEmpty()) { rcFile = QDir(directory()).filePath(mOptions.privateResource); if (changeFileExt(rcFile, DEV_PROJECT_EXT) == mFilename) { QFileInfo fileInfo(mFilename); rcFile = includeTrailingPathDelimiter(fileInfo.absolutePath()) + fileInfo.baseName() + "_private." + RC_EXT; } } else { QFileInfo fileInfo(mFilename); rcFile = includeTrailingPathDelimiter(fileInfo.absolutePath()) + fileInfo.baseName() + "_private." + RC_EXT; } rcFile = extractRelativePath(mFilename, rcFile); rcFile.replace(' ','_'); QStringList contents; contents.append("/* THIS FILE WILL BE OVERWRITTEN BY Red Panda C++ */"); contents.append("/* DO NOT EDIT! */"); contents.append(""); if (mOptions.includeVersionInfo) { contents.append("#include <windows.h> // include for version info constants"); contents.append(""); } foreach (const PProjectUnit& unit, mUnits) { if ( (getFileType(unit->fileName()) == FileType::WindowsResourceSource) && unit->compile() ) contents.append("#include \"" + extractRelativePath(directory(), unit->fileName()) + "\""); } if (!mOptions.icon.isEmpty()) { contents.append(""); QString icon = mOptions.icon; if (fileExists(icon)) { icon = extractRelativePath(mFilename, icon); icon.replace('\\', '/'); contents.append("A ICON \"" + icon + '"'); } else mOptions.icon = ""; } if (mOptions.supportXPThemes) { contents.append(""); contents.append("//"); contents.append("// SUPPORT FOR WINDOWS XP THEMES:"); contents.append("// THIS WILL MAKE THE PROGRAM USE THE COMMON CONTROLS"); contents.append("// LIBRARY VERSION 6.0 (IF IT IS AVAILABLE)"); contents.append("//"); if (!mOptions.exeOutput.isEmpty()) contents.append( "1 24 \"" + includeTrailingPathDelimiter(mOptions.exeOutput) + extractFileName(executable()) + ".Manifest\""); else contents.append("1 24 \"" + extractFileName(executable()) + ".Manifest\""); } if (mOptions.includeVersionInfo) { contents.append(""); contents.append("//"); contents.append("// TO CHANGE VERSION INFORMATION, EDIT PROJECT OPTIONS..."); contents.append("//"); contents.append("1 VERSIONINFO"); contents.append("FILEVERSION " + QString("%1,%2,%3,%4") .arg(mOptions.versionInfo.major) .arg(mOptions.versionInfo.minor) .arg(mOptions.versionInfo.release) .arg(mOptions.versionInfo.build)); contents.append("PRODUCTVERSION " + QString("%1,%2,%3,%4") .arg(mOptions.versionInfo.major) .arg(mOptions.versionInfo.minor) .arg(mOptions.versionInfo.release) .arg(mOptions.versionInfo.build)); switch(mOptions.type) { case ProjectType::GUI: case ProjectType::Console: contents.append("FILETYPE VFT_APP"); break; case ProjectType::StaticLib: contents.append("FILETYPE VFT_STATIC_LIB"); break; case ProjectType::DynamicLib: contents.append("FILETYPE VFT_DLL"); break; } contents.append("{"); contents.append(" BLOCK \"StringFileInfo\""); contents.append(" {"); contents.append(" BLOCK \"" + QString("%1%2") .arg(mOptions.versionInfo.languageID,4,16,QChar('0')) .arg(mOptions.versionInfo.charsetID,4,16,QChar('0')) + '"'); contents.append(" {"); contents.append(" VALUE \"CompanyName\", \"" + mOptions.versionInfo.companyName + "\""); contents.append(" VALUE \"FileVersion\", \"" + mOptions.versionInfo.fileVersion + "\""); contents.append(" VALUE \"FileDescription\", \"" + mOptions.versionInfo.fileDescription + "\""); contents.append(" VALUE \"InternalName\", \"" + mOptions.versionInfo.internalName + "\""); contents.append(" VALUE \"LegalCopyright\", \"" + mOptions.versionInfo.legalCopyright + '"'); contents.append(" VALUE \"LegalTrademarks\", \"" + mOptions.versionInfo.legalTrademarks + "\""); contents.append(" VALUE \"OriginalFilename\", \"" + mOptions.versionInfo.originalFilename + "\""); contents.append(" VALUE \"ProductName\", \"" + mOptions.versionInfo.productName + "\""); contents.append(" VALUE \"ProductVersion\", \"" + mOptions.versionInfo.productVersion + "\""); contents.append(" }"); contents.append(" }"); // additional block for windows 95->NT contents.append(" BLOCK \"VarFileInfo\""); contents.append(" {"); contents.append(" VALUE \"Translation\", " + QString("0x%1, %2") .arg(mOptions.versionInfo.languageID,4,16,QChar('0')) .arg(mOptions.versionInfo.charsetID)); contents.append(" }"); contents.append("}"); } rcFile = generateAbsolutePath(directory(),rcFile); if (contents.count() > 3) { stringsToFile(contents,rcFile); mOptions.privateResource = extractRelativePath(directory(), rcFile); } else { if (fileExists(rcFile)) QFile::remove(rcFile); QString resFile = changeFileExt(rcFile, RES_EXT); if (fileExists(resFile)) QFile::remove(resFile); mOptions.privateResource = ""; } // if fileExists(Res) then // FileSetDate(Res, DateTimeToFileDate(Now)); // fix the "Clock skew detected" warning ;) // create XP manifest if (mOptions.supportXPThemes) { QStringList content; content.append("<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"yes\"?>"); content.append("<assembly"); content.append(" xmlns=\"urn:schemas-microsoft-com:asm.v1\""); content.append(" manifestVersion=\"1.0\">"); content.append("<assemblyIdentity"); QString name = mName; name.replace(' ','_'); content.append(" name=\"DevCpp.Apps." + name + '\"'); content.append(" processorArchitecture=\"*\""); content.append(" version=\"1.0.0.0\""); content.append(" type=\"win32\"/>"); content.append("<description>" + name + "</description>"); content.append("<dependency>"); content.append(" <dependentAssembly>"); content.append(" <assemblyIdentity"); content.append(" type=\"win32\""); content.append(" name=\"Microsoft.Windows.Common-Controls\""); content.append(" version=\"6.0.0.0\""); content.append(" processorArchitecture=\"*\""); content.append(" publicKeyToken=\"6595b64144ccf1df\""); content.append(" language=\"*\""); content.append(" />"); content.append(" </dependentAssembly>"); content.append("</dependency>"); content.append("</assembly>"); stringsToFile(content,executable() + ".Manifest"); } else if (fileExists(executable() + ".Manifest")) QFile::remove(executable() + ".Manifest"); // create private header file QString hFile = changeFileExt(rcFile, H_EXT); contents.clear(); QString def = extractFileName(rcFile); def.replace(".","_"); contents.append("/* THIS FILE WILL BE OVERWRITTEN BY Red Panda C++ */"); contents.append("/* DO NOT EDIT ! */"); contents.append(""); contents.append("#ifndef " + def); contents.append("#define " + def); contents.append(""); contents.append("/* VERSION DEFINITIONS */"); contents.append("#define VER_STRING\t" + QString("\"%1.%2.%3.%4\"") .arg(mOptions.versionInfo.major) .arg(mOptions.versionInfo.minor) .arg(mOptions.versionInfo.release) .arg(mOptions.versionInfo.build)); contents.append(QString("#define VER_MAJOR\t%1").arg(mOptions.versionInfo.major)); contents.append(QString("#define VER_MINOR\t%1").arg(mOptions.versionInfo.minor)); contents.append(QString("#define VER_RELEASE\t%1").arg(mOptions.versionInfo.release)); contents.append(QString("#define VER_BUILD\t%1").arg(mOptions.versionInfo.build)); contents.append(QString("#define COMPANY_NAME\t\"%1\"") .arg(mOptions.versionInfo.companyName)); contents.append(QString("#define FILE_VERSION\t\"%1\"") .arg(mOptions.versionInfo.fileVersion)); contents.append(QString("#define FILE_DESCRIPTION\t\"%1\"") .arg(mOptions.versionInfo.fileDescription)); contents.append(QString("#define INTERNAL_NAME\t\"%1\"") .arg(mOptions.versionInfo.internalName)); contents.append(QString("#define LEGAL_COPYRIGHT\t\"%1\"") .arg(mOptions.versionInfo.legalCopyright)); contents.append(QString("#define LEGAL_TRADEMARKS\t\"%1\"") .arg(mOptions.versionInfo.legalTrademarks)); contents.append(QString("#define ORIGINAL_FILENAME\t\"%1\"") .arg(mOptions.versionInfo.originalFilename)); contents.append(QString("#define PRODUCT_NAME\t\"%1\"") .arg(mOptions.versionInfo.productName)); contents.append(QString("#define PRODUCT_VERSION\t\"%1\"") .arg(mOptions.versionInfo.productVersion)); contents.append(""); contents.append("#endif /*" + def + "*/"); stringsToFile(contents,hFile); } void Project::checkProjectFileForUpdate(SimpleIni &ini) { bool cnvt = false; int uCount = ini.GetLongValue("Project","UnitCount", 0); // check if using old way to store resources and fix it QString oldRes = QString::fromLocal8Bit(ini.GetValue("Project","Resources", "")); if (!oldRes.isEmpty()) { QFile::copy(mFilename,mFilename+".bak"); QStringList sl; sl = oldRes.split(';', #if QT_VERSION >= QT_VERSION_CHECK(5,15,0) Qt::SkipEmptyParts #else QString::SkipEmptyParts #endif ); for (int i=0;i<sl.count();i++){ const QString& s = sl[i]; QByteArray groupName = toByteArray(QString("Unit%1").arg(uCount+i)); ini.SetValue(groupName,"Filename", toByteArray(s)); ini.SetValue(groupName,"Folder", "Resources"); ini.SetLongValue(groupName,"Compile",true); } ini.SetLongValue("Project","UnitCount",uCount+sl.count()); QString folders = QString::fromLocal8Bit(ini.GetValue("Project","Folders","")); if (!folders.isEmpty()) folders += ",Resources"; else folders = "Resources"; ini.SetValue("Project","Folders",toByteArray(folders)); cnvt = true; ini.Delete("Project","Resources"); ini.Delete("Project","Focused"); ini.Delete("Project","Order"); ini.Delete("Project","DebugInfo"); ini.Delete("Project","ProfileInfo"); ini.SaveFile(mFilename.toLocal8Bit()); } if (cnvt) QMessageBox::information( nullptr, tr("Project Updated"), tr("Your project was succesfully updated to a newer file format!") +"<br />" +tr("If something has gone wrong, we kept a backup-file: '%1'...") .arg(mFilename+".bak"), QMessageBox::Ok); } void Project::closeUnit(PProjectUnit& unit) { saveLayout(); Editor * editor = unitEditor(unit); if (editor) { mEditorList->forceCloseEditor(editor); } } void Project::createFolderNodes() { for (int idx=0;idx<mFolders.count();idx++) { PProjectModelNode node = mRootNode; QString s = mFolders[idx]; int i = s.indexOf('/'); while (i>=0) { PProjectModelNode findnode; for (int c=0;c<node->children.count();c++) { if (node->children[c]->text == s.mid(0,i)) findnode = node->children[c]; } if (!findnode) node = makeNewFolderNode(s.mid(0,i),node); else node = findnode; if (!node->isUnit) { qDebug()<<"node "<<node->text<<"is not a folder:"<<s; node = mRootNode; } s.remove(0,i+1); i = s.indexOf('/'); } node = makeNewFolderNode(s, node); mCustomFolderNodes.append(node); } } static void addFolderRecursively(QSet<QString>& folders, QString folder) { if (folder.isEmpty()) return; folders.insert(excludeTrailingPathDelimiter(folder)); QString parentFolder = QFileInfo(folder).absolutePath(); if (parentFolder==folder) return; addFolderRecursively(folders, parentFolder); } void Project::createFileSystemFolderNodes() { QSet<QString> headerFolders; QSet<QString> sourceFolders; QSet<QString> otherFolders; mRootNode->children.clear(); mSpecialNodes.clear(); mFileSystemFolderNodes.clear(); foreach (const PProjectUnit& unit, mUnits) { QFileInfo fileInfo(unit->fileName()); if (isHFile(fileInfo.fileName())) { addFolderRecursively(headerFolders,fileInfo.absolutePath()); } else if (isCFile(fileInfo.fileName())) { addFolderRecursively(sourceFolders,fileInfo.absolutePath()); } else { addFolderRecursively(otherFolders,fileInfo.absolutePath()); } } PProjectModelNode node = makeNewFolderNode(tr("Headers"), mRootNode, ProjectModelNodeType::DUMMY_HEADERS_FOLDER, 1000); createFileSystemFolderNode(ProjectModelNodeType::DUMMY_HEADERS_FOLDER,folder(),node, headerFolders); mCustomFolderNodes.append(node); mSpecialNodes.insert(ProjectModelNodeType::DUMMY_HEADERS_FOLDER,node); node = makeNewFolderNode(tr("Sources"), mRootNode, ProjectModelNodeType::DUMMY_SOURCES_FOLDER, 900); createFileSystemFolderNode(ProjectModelNodeType::DUMMY_SOURCES_FOLDER,folder(),node, sourceFolders); mCustomFolderNodes.append(node); mSpecialNodes.insert(ProjectModelNodeType::DUMMY_SOURCES_FOLDER,node); node = makeNewFolderNode(tr("Others"), mRootNode, ProjectModelNodeType::DUMMY_OTHERS_FOLDER, 800); createFileSystemFolderNode(ProjectModelNodeType::DUMMY_OTHERS_FOLDER,folder(),node, otherFolders); mCustomFolderNodes.append(node); mSpecialNodes.insert(ProjectModelNodeType::DUMMY_OTHERS_FOLDER,node); } void Project::createFileSystemFolderNode( ProjectModelNodeType folderType, const QString &folderName, PProjectModelNode parent, const QSet<QString>& validFolders) { QDirIterator iter(folderName); while (iter.hasNext()) { iter.next(); QFileInfo fileInfo = iter.fileInfo(); if (fileInfo.isHidden() || fileInfo.fileName().startsWith('.')) continue; if (fileInfo.isDir() && validFolders.contains(fileInfo.absoluteFilePath())) { PProjectModelNode node = makeNewFolderNode(fileInfo.fileName(),parent); mFileSystemFolderNodes.insert(QString("%1/%2").arg((int)folderType).arg(fileInfo.absoluteFilePath()),node); createFileSystemFolderNode(folderType,fileInfo.absoluteFilePath(), node, validFolders); } } } PProjectUnit Project::doAutoOpen() { QHash<QString,PProjectEditorLayout> layouts = loadLayout(); QHash<int,PProjectEditorLayout> opennedMap; QString focusedFilename; foreach (const PProjectEditorLayout &layout,layouts) { if (layout->isOpen && layout->order>=0) { if (layout->isFocused) focusedFilename = layout->filename; opennedMap.insert(layout->order,layout); } } for (int i=0;i<mUnits.count();i++) { PProjectEditorLayout editorLayout = opennedMap.value(i,PProjectEditorLayout()); if (editorLayout) { PProjectUnit unit = findUnit(editorLayout->filename); openUnit(unit,editorLayout); } } if (!focusedFilename.isEmpty()) { PProjectUnit unit = findUnit(focusedFilename); if (unit) { Editor * editor = unitEditor(unit); if (editor) editor->activate(); } return unit; } return PProjectUnit(); } bool Project::fileAlreadyExists(const QString &s) { foreach (const PProjectUnit& unit, mUnits) { if (unit->fileName() == s) return true; } return false; } PProjectModelNode Project::findFileSystemFolderNode(const QString &folderPath, ProjectModelNodeType nodeType) { PProjectModelNode node = mFileSystemFolderNodes.value(QString("%1/%2").arg((int)nodeType).arg(folderPath), PProjectModelNode()); if (node) return node; PProjectModelNode parentNode = mSpecialNodes.value(nodeType,PProjectModelNode()); if (parentNode) { QString projectFolder = includeTrailingPathDelimiter(directory()); if (folderPath.startsWith(projectFolder)) { QString pathStr = folderPath.mid(projectFolder.length()); QStringList paths = pathStr.split("/"); PProjectModelNode currentParentNode = parentNode; QString currentFolderFullPath=directory(); for (int i=0;i<paths.length();i++) { QString currentFolderName = paths[i]; currentFolderFullPath = currentFolderFullPath+"/"+currentFolderName; bool found=false; foreach(PProjectModelNode tempNode, parentNode->children) { if (tempNode->folderNodeType == ProjectModelNodeType::Folder && tempNode->text == currentFolderName) { found=true; currentParentNode = tempNode; break; } } if (!found) { PProjectModelNode newNode = makeNewFolderNode(currentFolderName,currentParentNode); mFileSystemFolderNodes.insert(QString("%1/%2").arg((int)nodeType).arg(currentFolderFullPath),newNode); currentParentNode = newNode; } } return currentParentNode; } return parentNode; } return mRootNode; } PProjectModelNode Project::getCustomeFolderNodeFromName(const QString &name) { int index = mFolders.indexOf(name); if (index>=0) { return mCustomFolderNodes[index]; } return mRootNode; } QString Project::getNodePath(PProjectModelNode node) { QString result; if (!node) return result; if (node->isUnit) // not a folder return result; PProjectModelNode p = node; while (p && !p->isUnit && p!=mRootNode) { if (!result.isEmpty()) result = p->text + "/" + result; else result = p->text; p = p->parent.lock(); } return result; } PProjectModelNode Project::getParentFileSystemFolderNode(const QString &filename) { QFileInfo fileInfo(filename); ProjectModelNodeType folderNodeType; if (isHFile(fileInfo.fileName()) && !fileInfo.suffix().isEmpty()) { folderNodeType = ProjectModelNodeType::DUMMY_HEADERS_FOLDER; } else if (isCFile(fileInfo.fileName())) { folderNodeType = ProjectModelNodeType::DUMMY_SOURCES_FOLDER; } else { folderNodeType = ProjectModelNodeType::DUMMY_OTHERS_FOLDER; } return findFileSystemFolderNode(fileInfo.absolutePath(),folderNodeType); } void Project::incrementBuildNumber() { mOptions.versionInfo.build++; mOptions.versionInfo.fileVersion = QString("%1.%2.%3.%4") .arg(mOptions.versionInfo.major) .arg(mOptions.versionInfo.minor) .arg(mOptions.versionInfo.release) .arg(mOptions.versionInfo.build); if (mOptions.versionInfo.syncProduct) mOptions.versionInfo.productVersion = mOptions.versionInfo.fileVersion; saveOptions(); } QHash<QString, PProjectEditorLayout> Project::loadLayout() { QHash<QString,PProjectEditorLayout> layouts; QString jsonFilename = changeFileExt(filename(), "layout"); QFile file(jsonFilename); if (!file.open(QIODevice::ReadOnly)) return layouts; QByteArray content = file.readAll(); QJsonParseError parseError; QJsonDocument doc(QJsonDocument::fromJson(content,&parseError)); file.close(); if (parseError.error!=QJsonParseError::NoError || !doc.isArray()) return layouts; QJsonArray jsonLayouts=doc.array(); for (int i=0;i<jsonLayouts.size();i++) { QJsonObject jsonLayout = jsonLayouts[i].toObject(); QString unitFilename = jsonLayout["filename"].toString(); if (mUnits.contains(unitFilename)) { PProjectEditorLayout editorLayout = std::make_shared<ProjectEditorLayout>(); editorLayout->filename=unitFilename; editorLayout->topLine=jsonLayout["topLine"].toInt(); editorLayout->left=jsonLayout["left"].toInt(); editorLayout->caretX=jsonLayout["caretX"].toInt(); editorLayout->caretY=jsonLayout["caretY"].toInt(); editorLayout->order=jsonLayout["order"].toInt(-1); editorLayout->isFocused=jsonLayout["focused"].toBool(); editorLayout->isOpen=jsonLayout["isOpen"].toBool(); layouts.insert(unitFilename,editorLayout); } } return layouts; } void Project::loadOptions(SimpleIni& ini) { mName = fromByteArray(ini.GetValue("Project","name", "")); QString icon = fromByteArray(ini.GetValue("Project", "icon", "")); if (icon.isEmpty()) { mOptions.icon = ""; } else { mOptions.icon = generateAbsolutePath(directory(),icon); } mOptions.version = ini.GetLongValue("Project", "Ver", 0); if (mOptions.version > 0) { // ver > 0 is at least a v5 project if (mOptions.version < 3) { mOptions.version = 3; QMessageBox::information(nullptr, tr("Settings need update"), tr("The compiler settings format of Red Panda C++ has changed.") +"<BR /><BR />" +tr("Please update your settings at Project >> Project Options >> Compiler and save your project."), QMessageBox::Ok); } mOptions.type = static_cast<ProjectType>(ini.GetLongValue("Project", "type", 0)); // ;CONFIG_LINE; is used in olded version config files (<2.17) // keep it for compatibility mOptions.compilerCmd = fromByteArray(ini.GetValue("Project", "Compiler", "")).replace(";CONFIG_LINE;","\n"); mOptions.cppCompilerCmd = fromByteArray(ini.GetValue("Project", "CppCompiler", "")).replace(";CONFIG_LINE;","\n"); mOptions.linkerCmd = fromByteArray(ini.GetValue("Project", "Linker", "")).replace(";CONFIG_LINE;","\n"); mOptions.resourceCmd = fromByteArray(ini.GetValue("Project", "ResourceCommand", "")).replace(";CONFIG_LINE;","\n"); mOptions.binDirs = absolutePaths(fromByteArray(ini.GetValue("Project", "Bins", "")).split(";", #if QT_VERSION >= QT_VERSION_CHECK(5,15,0) Qt::SkipEmptyParts #else QString::SkipEmptyParts #endif )); mOptions.libDirs = absolutePaths(fromByteArray(ini.GetValue("Project", "Libs", "")).split(";", #if QT_VERSION >= QT_VERSION_CHECK(5,15,0) Qt::SkipEmptyParts #else QString::SkipEmptyParts #endif )); mOptions.includeDirs = absolutePaths(fromByteArray(ini.GetValue("Project", "Includes", "")).split(";", #if QT_VERSION >= QT_VERSION_CHECK(5,15,0) Qt::SkipEmptyParts #else QString::SkipEmptyParts #endif )); mOptions.privateResource = fromByteArray(ini.GetValue("Project", "PrivateResource", "")); mOptions.resourceIncludes = absolutePaths(fromByteArray(ini.GetValue("Project", "ResourceIncludes", "")).split(";", #if QT_VERSION >= QT_VERSION_CHECK(5,15,0) Qt::SkipEmptyParts #else QString::SkipEmptyParts #endif )); mOptions.makeIncludes = absolutePaths(fromByteArray(ini.GetValue("Project", "MakeIncludes", "")).split(";", #if QT_VERSION >= QT_VERSION_CHECK(5,15,0) Qt::SkipEmptyParts #else QString::SkipEmptyParts #endif )); mOptions.isCpp = ini.GetBoolValue("Project", "IsCpp", false); mOptions.exeOutput = generateAbsolutePath(directory(), fromByteArray(ini.GetValue("Project", "ExeOutput", ""))); mOptions.objectOutput = generateAbsolutePath(directory(), fromByteArray(ini.GetValue("Project", "ObjectOutput", ""))); mOptions.logOutput = generateAbsolutePath(directory(), fromByteArray(ini.GetValue("Project", "LogOutput", ""))); mOptions.logOutputEnabled = ini.GetBoolValue("Project", "LogOutputEnabled", false); mOptions.overrideOutput = ini.GetBoolValue("Project", "OverrideOutput", false); mOptions.overridenOutput = fromByteArray(ini.GetValue("Project", "OverrideOutputName", "")); mOptions.hostApplication = generateAbsolutePath(directory(), fromByteArray(ini.GetValue("Project", "HostApplication", ""))); mOptions.useCustomMakefile = ini.GetBoolValue("Project", "UseCustomMakefile", false); mOptions.customMakefile = generateAbsolutePath(directory(),fromByteArray(ini.GetValue("Project", "CustomMakefile", ""))); mOptions.usePrecompiledHeader = ini.GetBoolValue("Project", "UsePrecompiledHeader", false); mOptions.precompiledHeader = generateAbsolutePath(directory(),fromByteArray(ini.GetValue("Project", "PrecompiledHeader", ""))); mOptions.cmdLineArgs = fromByteArray(ini.GetValue("Project", "CommandLine", "")); mFolders = fromByteArray(ini.GetValue("Project", "Folders", "")).split(";", #if QT_VERSION >= QT_VERSION_CHECK(5,15,0) Qt::SkipEmptyParts #else QString::SkipEmptyParts #endif ); mOptions.includeVersionInfo = ini.GetBoolValue("Project", "IncludeVersionInfo", false); mOptions.supportXPThemes = ini.GetBoolValue("Project", "SupportXPThemes", false); mOptions.compilerSet = ini.GetLongValue("Project", "CompilerSet", pSettings->compilerSets().defaultIndex()); mOptions.modelType = (ProjectModelType)ini.GetLongValue("Project", "ModelType", (int)ProjectModelType::Custom); mOptions.classBrowserType = (ProjectClassBrowserType)ini.GetLongValue("Project", "ClassBrowserType", (int)ProjectClassBrowserType::CurrentFile); if (mOptions.compilerSet >= (int)pSettings->compilerSets().size() || mOptions.compilerSet < 0) { // TODO: change from indices to names QMessageBox::critical( nullptr, tr("Compiler not found"), tr("The compiler set you have selected for this project, no longer exists.") +"<BR />" +tr("It will be substituted by the global compiler set."), QMessageBox::Ok ); setCompilerSet(pSettings->compilerSets().defaultIndex()); saveOptions(); } Settings::PCompilerSet pSet = pSettings->compilerSets().getSet(mOptions.compilerSet); if (pSet) { QByteArray oldCompilerOptions = ini.GetValue("Project", "CompilerSettings", ""); if (!oldCompilerOptions.isEmpty()) { //version 2 compatibility // test if it is created by old dev-c++ SimpleIni::TNamesDepend oKeys; ini.GetAllKeys("Project", oKeys); bool isNewDev=false; for(const SimpleIni::Entry& entry:oKeys) { QString key(entry.pItem); if (key=="UsePrecompiledHeader" || key == "CompilerSetType" || key == "StaticLink" || key == "AddCharset" || key == "ExecEncoding" || key == "Encoding" || key == "UseUTF8") { isNewDev = true; break; } } if (!isNewDev && oldCompilerOptions.length()>=25) { char t = oldCompilerOptions[18]; oldCompilerOptions[18]=oldCompilerOptions[21]; oldCompilerOptions[21]=t; } for (int i=0;i<oldCompilerOptions.length();i++) { QString key = pSettings->compilerSets().getKeyFromCompilerCompatibleIndex(i); PCompilerOption pOption = CompilerInfoManager::getCompilerOption( pSet->compilerType(), key); if (pOption) { int val = Settings::CompilerSet::charToValue(oldCompilerOptions[i]); if (pOption->choices.isEmpty()) { if (val>0) mOptions.compilerOptions.insert(key,COMPILER_OPTION_ON); else mOptions.compilerOptions.insert(key,""); } else { if (val>0 && val <= pOption->choices.length()) mOptions.compilerOptions.insert(key,pOption->choices[val-1].second); else mOptions.compilerOptions.insert(key,""); } } } } else { //version 3 SimpleIni::TNamesDepend oKeys; ini.GetAllKeys("CompilerSettings", oKeys); for(const SimpleIni::Entry& entry:oKeys) { QString key(entry.pItem); mOptions.compilerOptions.insert( key, ini.GetValue("CompilerSettings", entry.pItem, "")); } } } mOptions.staticLink = ini.GetBoolValue("Project", "StaticLink", true); mOptions.execEncoding = ini.GetValue("Project","ExecEncoding", ENCODING_SYSTEM_DEFAULT); mOptions.addCharset = ini.GetBoolValue("Project", "AddCharset", true); bool useUTF8 = ini.GetBoolValue("Project", "UseUTF8", false); if (useUTF8) { mOptions.encoding = ini.GetValue("Project","Encoding", ENCODING_UTF8); } else { mOptions.encoding = ini.GetValue("Project","Encoding", pSettings->editor().defaultEncoding()); } if (mOptions.encoding == ENCODING_AUTO_DETECT) mOptions.encoding = pSettings->editor().defaultEncoding(); if (mOptions.encoding == ENCODING_AUTO_DETECT) mOptions.encoding = ENCODING_SYSTEM_DEFAULT; mOptions.allowParallelBuilding = ini.GetBoolValue("Project","AllowParallelBuilding"); mOptions.parellelBuildingJobs = ini.GetLongValue("Project","ParellelBuildingJobs"); mOptions.versionInfo.major = ini.GetLongValue("VersionInfo", "Major", 0); mOptions.versionInfo.minor = ini.GetLongValue("VersionInfo", "Minor", 1); mOptions.versionInfo.release = ini.GetLongValue("VersionInfo", "Release", 1); mOptions.versionInfo.build = ini.GetLongValue("VersionInfo", "Build", 1); mOptions.versionInfo.languageID = ini.GetLongValue("VersionInfo", "LanguageID", 0x0409); mOptions.versionInfo.charsetID = ini.GetLongValue("VersionInfo", "CharsetID", 0x04E4); mOptions.versionInfo.companyName = fromByteArray(ini.GetValue("VersionInfo", "CompanyName", "")); mOptions.versionInfo.fileVersion = fromByteArray(ini.GetValue("VersionInfo", "FileVersion", "0.1")); mOptions.versionInfo.fileDescription = fromByteArray(ini.GetValue("VersionInfo", "FileDescription", toByteArray(tr("Developed using the Red Panda C++ IDE")))); mOptions.versionInfo.internalName = fromByteArray(ini.GetValue("VersionInfo", "InternalName", "")); mOptions.versionInfo.legalCopyright = fromByteArray(ini.GetValue("VersionInfo", "LegalCopyright", "")); mOptions.versionInfo.legalTrademarks = fromByteArray(ini.GetValue("VersionInfo", "LegalTrademarks", "")); mOptions.versionInfo.originalFilename = fromByteArray(ini.GetValue("VersionInfo", "OriginalFilename", toByteArray(extractFileName(executable())))); mOptions.versionInfo.productName = fromByteArray(ini.GetValue("VersionInfo", "ProductName", toByteArray(mName))); mOptions.versionInfo.productVersion = fromByteArray(ini.GetValue("VersionInfo", "ProductVersion", "0.1.1.1")); mOptions.versionInfo.autoIncBuildNr = ini.GetBoolValue("VersionInfo", "AutoIncBuildNr", false); mOptions.versionInfo.syncProduct = ini.GetBoolValue("VersionInfo", "SyncProduct", false); } } void Project::loadUnitLayout(Editor *e) { if (!e) return; QHash<QString, PProjectEditorLayout> layouts = loadLayout(); PProjectEditorLayout layout = layouts.value(e->filename(),PProjectEditorLayout()); if (layout) { e->setCaretY(layout->caretY); e->setCaretX(layout->caretX); e->setTopLine(layout->topLine); e->setLeftPos(layout->left); } } QString Project::relativePath(const QString &filename) { QString appPath = includeTrailingPathDelimiter(pSettings->dirs().appDir()); QString projectPath = includeTrailingPathDelimiter(directory()); if (filename.startsWith(appPath) && !filename.startsWith(projectPath)) { return "%APP_PATH%/"+filename.mid(appPath.length()); } QDir projectDir(directory()); QDir grandparentDir(projectDir.absoluteFilePath("../../")); QString grandparentPath=grandparentDir.absolutePath(); if (grandparentDir.exists() && filename.startsWith(grandparentPath)) return extractRelativePath(directory(),filename); return filename; } QStringList Project::relativePaths(const QStringList &files) { QStringList lst; foreach(const QString& file,files) { lst.append(relativePath(file)); } return lst; } QString Project::absolutePath(const QString &filename) { QString appSuffix = "%APP_PATH%/"; if (filename.startsWith(appSuffix)) { return includeTrailingPathDelimiter(pSettings->dirs().appDir()) + filename.mid(appSuffix.length()); } return generateAbsolutePath(directory(),filename); } QStringList Project::absolutePaths(const QStringList &files) { QStringList lst; foreach(const QString& file,files) { lst.append(absolutePath(file)); } return lst; } PCppParser Project::cppParser() { return mParser; } void Project::removeFolderRecurse(PProjectModelNode node) { if (!node) return ; // Recursively remove folders for (int i=node->children.count()-1;i>=0;i++) { PProjectModelNode childNode = node->children[i]; // Remove folder inside folder if (!childNode->isUnit && childNode->level>0) { removeFolderRecurse(childNode); // Or remove editors at this level } else if (childNode->isUnit && childNode->level > 0) { // Remove editor in folder from project PProjectUnit unit = childNode->pUnit.lock(); if (!removeUnit(unit,true)) return; } } PProjectModelNode parent = node->parent.lock(); if (parent) { parent->children.removeAll(node); } } void Project::updateFolderNode(PProjectModelNode node) { for (int i=0;i<node->children.count();i++){ PProjectModelNode child = node->children[i]; if (!child->isUnit) { mFolders.append(getNodePath(child)); updateFolderNode(child); } } } void Project::updateCompilerSetting() { Settings::PCompilerSet defaultSet = pSettings->compilerSets().getSet(mOptions.compilerSet); if (defaultSet) { mOptions.staticLink = defaultSet->staticLink(); mOptions.compilerOptions = defaultSet->compileOptions(); } else { mOptions.staticLink = false; } } QFileSystemWatcher *Project::fileSystemWatcher() const { return mFileSystemWatcher; } QString Project::fileSystemNodeFolderPath(const PProjectModelNode &node) { QString result; if (node != mRootNode) { PProjectModelNode pNode = node; while (pNode && pNode->folderNodeType == ProjectModelNodeType::Folder) { result = node->text + "/" +result; pNode = pNode->parent.lock(); } } result = folder() + "/" + result; return result; } QStringList Project::binDirs() { QStringList lst = options().binDirs; Settings::PCompilerSet compilerSet = pSettings->compilerSets().getSet(options().compilerSet); if (compilerSet) { lst.append(compilerSet->binDirs()); } return lst; } void Project::renameFolderNode(PProjectModelNode node, const QString newName) { if (!node) return; if (node->isUnit) return; node->text = newName; updateFolders(); setModified(true); emit nodeRenamed(); } EditorList *Project::editorList() const { return mEditorList; } ProjectModelType Project::modelType() const { return mOptions.modelType; } void Project::setModelType(ProjectModelType type) { if (type!=mOptions.modelType) { mOptions.modelType = type; rebuildNodes(); } } ProjectOptions &Project::options() { return mOptions; } ProjectModel *Project::model() { return &mModel; } const PProjectModelNode &Project::rootNode() const { return mRootNode; } const QString &Project::name() const { return mName; } void Project::setName(const QString &newName) { if (newName != mName) { mName = newName; mRootNode->text = newName; setModified(true); } } const QString &Project::filename() const { return mFilename; } ProjectUnit::ProjectUnit(Project* parent) { mNode = nullptr; mParent = parent; // mFileMissing = false; mPriority=0; mNew = true; mEncoding=ENCODING_PROJECT; mRealEncoding=""; } Project *ProjectUnit::parent() const { return mParent; } const QString &ProjectUnit::fileName() const { return mFileName; } void ProjectUnit::setFileName(QString newFileName) { newFileName = QFileInfo(newFileName).absoluteFilePath(); if (mFileName != newFileName) { mFileName = newFileName; if (mNode) { mNode->text = extractFileName(mFileName); } } } void ProjectUnit::setNew(bool newNew) { mNew = newNew; } const QByteArray &ProjectUnit::realEncoding() const { return mRealEncoding; } void ProjectUnit::setRealEncoding(const QByteArray &newRealEncoding) { mRealEncoding = newRealEncoding; } const QString &ProjectUnit::folder() const { return mFolder; } void ProjectUnit::setFolder(const QString &newFolder) { mFolder = newFolder; } bool ProjectUnit::compile() const { return mCompile; } void ProjectUnit::setCompile(bool newCompile) { mCompile = newCompile; } bool ProjectUnit::compileCpp() const { return mCompileCpp; } void ProjectUnit::setCompileCpp(bool newCompileCpp) { mCompileCpp = newCompileCpp; } bool ProjectUnit::overrideBuildCmd() const { return mOverrideBuildCmd; } void ProjectUnit::setOverrideBuildCmd(bool newOverrideBuildCmd) { mOverrideBuildCmd = newOverrideBuildCmd; } const QString &ProjectUnit::buildCmd() const { return mBuildCmd; } void ProjectUnit::setBuildCmd(const QString &newBuildCmd) { mBuildCmd = newBuildCmd; } bool ProjectUnit::link() const { return mLink; } void ProjectUnit::setLink(bool newLink) { mLink = newLink; } int ProjectUnit::priority() const { return mPriority; } void ProjectUnit::setPriority(int newPriority) { if (mPriority!=newPriority) { mPriority = newPriority; if (mNode) mNode->priority = mPriority; } } const QByteArray &ProjectUnit::encoding() const { return mEncoding; } void ProjectUnit::setEncoding(const QByteArray &newEncoding) { if (mEncoding != newEncoding) { Editor * editor=mParent->unitEditor(this); if (editor) { editor->setEncodingOption(newEncoding); } mEncoding = newEncoding; } } PProjectModelNode &ProjectUnit::node() { return mNode; } void ProjectUnit::setNode(const PProjectModelNode &newNode) { mNode = newNode; } //bool ProjectUnit::FileMissing() const //{ // return mFileMissing; //} //void ProjectUnit::setFileMissing(bool newDontSave) //{ // mFileMissing = newDontSave; //} ProjectModel::ProjectModel(Project *project, QObject *parent): QAbstractItemModel(parent), mProject(project) { mUpdateCount = 0; //delete in the destructor mIconProvider = new CustomFileIconProvider(); } ProjectModel::~ProjectModel() { delete mIconProvider; } void ProjectModel::beginUpdate() { if (mUpdateCount==0) { beginResetModel(); } mUpdateCount++; } void ProjectModel::endUpdate() { mUpdateCount--; if (mUpdateCount==0) { mIconProvider->setRootFolder(mProject->folder()); endResetModel(); } } CustomFileIconProvider *ProjectModel::iconProvider() const { return mIconProvider; } bool ProjectModel::insertRows(int row, int count, const QModelIndex &parent) { beginInsertRows(parent,row,row+count-1); endInsertRows(); return true; } bool ProjectModel::removeRows(int row, int count, const QModelIndex &parent) { beginRemoveRows(parent,row,row+count-1); if (!parent.isValid()) return false; ProjectModelNode* parentNode = static_cast<ProjectModelNode*>(parent.internalPointer()); if (!parentNode) return false; parentNode->children.removeAt(row); endRemoveRows(); return true; } Project *ProjectModel::project() const { return mProject; } QModelIndex ProjectModel::index(int row, int column, const QModelIndex &parent) const { if (!parent.isValid()) { return createIndex(row,column,mProject->rootNode().get()); } ProjectModelNode* parentNode = static_cast<ProjectModelNode*>(parent.internalPointer()); if (!parentNode) { return QModelIndex(); } if (row<0 || row>=parentNode->children.count()) return QModelIndex(); return createIndex(row,column,parentNode->children[row].get()); } QModelIndex ProjectModel::parent(const QModelIndex &child) const { if (!child.isValid()) return QModelIndex(); ProjectModelNode * node = static_cast<ProjectModelNode*>(child.internalPointer()); if (!node) return QModelIndex(); return getParentIndex(node); } int ProjectModel::rowCount(const QModelIndex &parent) const { if (!parent.isValid()) return 1; ProjectModelNode* p = static_cast<ProjectModelNode*>(parent.internalPointer()); if (p) { return p->children.count(); } else { return mProject->rootNode()->children.count(); } } int ProjectModel::columnCount(const QModelIndex &) const { return 1; } QVariant ProjectModel::data(const QModelIndex &index, int role) const { if (!index.isValid()) return QVariant(); ProjectModelNode* p = static_cast<ProjectModelNode*>(index.internalPointer()); if (!p) return QVariant(); if (role == Qt::DisplayRole) { #ifdef ENABLE_VCS if (p == mProject->rootNode().get()) { QString branch; if (mIconProvider->VCSRepository()->hasRepository(branch)) return QString("%1 [%2]").arg(p->text,branch); } #endif return p->text; } else if (role==Qt::EditRole) { return p->text; } else if (role == Qt::DecorationRole) { QIcon icon; if (p->isUnit) { PProjectUnit unit = p->pUnit.lock(); if (unit) icon = mIconProvider->icon(unit->fileName()); } else { if (p == mProject->rootNode().get()) { #ifdef ENABLE_VCS QString branch; if (mIconProvider->VCSRepository()->hasRepository(branch)) icon = pIconsManager->getIcon(IconsManager::FILESYSTEM_GIT); #endif } else { switch(p->folderNodeType) { case ProjectModelNodeType::DUMMY_HEADERS_FOLDER: icon = pIconsManager->getIcon(IconsManager::FILESYSTEM_HEADERS_FOLDER); break; case ProjectModelNodeType::DUMMY_SOURCES_FOLDER: icon = pIconsManager->getIcon(IconsManager::FILESYSTEM_SOURCES_FOLDER); break; default: icon = pIconsManager->getIcon(IconsManager::FILESYSTEM_FOLDER); } } if (icon.isNull()) icon = mIconProvider->icon(QFileIconProvider::Folder); } return icon; } return QVariant(); } Qt::ItemFlags ProjectModel::flags(const QModelIndex &index) const { if (!index.isValid()) return Qt::NoItemFlags; ProjectModelNode* p = static_cast<ProjectModelNode*>(index.internalPointer()); if (!p) return Qt::NoItemFlags; if (p==mProject->rootNode().get()) return Qt::ItemIsEnabled | Qt::ItemIsDropEnabled | Qt::ItemIsEditable; if (mProject && mProject->modelType() == ProjectModelType::FileSystem) { Qt::ItemFlags flags = Qt::ItemIsEnabled | Qt::ItemIsSelectable; if (p->isUnit) flags.setFlag(Qt::ItemIsEditable); return flags; } else { Qt::ItemFlags flags = Qt::ItemIsEnabled | Qt::ItemIsEditable | Qt::ItemIsSelectable | Qt::ItemIsDragEnabled; if (!p->isUnit) { flags.setFlag(Qt::ItemIsDropEnabled); flags.setFlag(Qt::ItemIsDragEnabled,false); } return flags; } } bool ProjectModel::setData(const QModelIndex &index, const QVariant &value, int role) { if (!index.isValid()) return false; ProjectModelNode* p = static_cast<ProjectModelNode*>(index.internalPointer()); PProjectModelNode node = mProject->pointerToNode(p); if (!node) return false; if (role == Qt::EditRole) { if (node == mProject->rootNode()) { QString newName = value.toString().trimmed(); if (newName.isEmpty()) return false; mProject->setName(newName); emit dataChanged(index,index); return true; } PProjectUnit unit = node->pUnit.lock(); if (unit) { //change unit name QString newName = value.toString().trimmed(); if (newName.isEmpty()) return false; if (newName == node->text) return false; QString oldName = unit->fileName(); QString curDir = extractFilePath(oldName); newName = generateAbsolutePath(curDir,newName); // Only continue if the user says so... if (fileExists(newName) && newName.compare(oldName, PATH_SENSITIVITY)!=0) { // don't remove when changing case for example if (QMessageBox::question(nullptr, tr("File exists"), tr("File '%1' already exists. Delete it now?") .arg(newName), QMessageBox::Yes | QMessageBox::No, QMessageBox::No) == QMessageBox::Yes) { // Close the target file... Editor * e=mProject->editorList()->getOpenedEditorByFilename(newName); if (e) mProject->editorList()->closeEditor(e); // Remove it from the current project... PProjectUnit unit = mProject->findUnit(newName); if (unit) { mProject->removeUnit(unit,false); } // All references to the file are removed. Delete the file from disk if (!QFile::remove(newName)) { QMessageBox::critical(nullptr, tr("Remove failed"), tr("Failed to remove file '%1'") .arg(newName), QMessageBox::Ok); return false; } } else { return false; } } // Target filename does not exist anymore. Do a rename // change name in project file first (no actual file renaming on disk) //save old file, if it is opened; // remove old file from monitor list mProject->fileSystemWatcher()->removePath(oldName); if (!QFile::rename(oldName,newName)) { QMessageBox::critical(nullptr, tr("Rename failed"), tr("Failed to rename file '%1' to '%2'") .arg(oldName,newName), QMessageBox::Ok); return false; } mProject->renameUnit(unit,newName); // Add new filename to file minitor mProject->fileSystemWatcher()->addPath(newName); mProject->saveAll(); return true; } else { //change folder name QString newName = value.toString().trimmed(); if (newName.isEmpty()) return false; if (newName == node->text) return false; mProject->renameFolderNode(node,newName); emit dataChanged(index,index); mProject->saveAll(); return true; } } return false; } void ProjectModel::refreshIcon(const QModelIndex &index, bool update) { if (!index.isValid()) return; if (update) mIconProvider->update(); QVector<int> roles; roles.append(Qt::DecorationRole); emit dataChanged(index,index, roles); } void ProjectModel::refreshIcon(const QString &filename) { PProjectUnit unit=mProject->findUnit(filename); if (!unit) return; PProjectModelNode node=unit->node(); QModelIndex index = getNodeIndex(node.get()); refreshIcon(index); } void ProjectModel::refreshIcons() { mIconProvider->update(); mProject->rootNode(); } void ProjectModel::refreshNodeIconRecursive(PProjectModelNode node) { QModelIndex index=getNodeIndex(node.get()); refreshIcon(index,false); foreach( PProjectModelNode child, node->children) { refreshNodeIconRecursive(child); } } QModelIndex ProjectModel::getNodeIndex(ProjectModelNode *node) const { if (!node) return QModelIndex(); PProjectModelNode parent = node->parent.lock(); if (!parent) // root node return createIndex(0,0,node); int row = -1; for (int i=0;i<parent->children.count();i++) { const PProjectModelNode& pNode=parent->children[i]; if (pNode.get()==node) { row = i; } } if (row<0) return QModelIndex(); return createIndex(row,0,node); } QModelIndex ProjectModel::getParentIndex(ProjectModelNode * node) const { PProjectModelNode parent = node->parent.lock(); if (!parent) // root node return QModelIndex(); PProjectModelNode grand = parent->parent.lock(); if (!grand) { return createIndex(0,0,parent.get()); } int row = grand->children.indexOf(parent); if (row<0) return QModelIndex(); return createIndex(row,0,parent.get()); } QModelIndex ProjectModel::rootIndex() const { return getNodeIndex(mProject->rootNode().get()); } bool ProjectModel::canDropMimeData(const QMimeData * data, Qt::DropAction action, int /*row*/, int /*column*/, const QModelIndex &parent) const { if (!data || action != Qt::MoveAction) return false; if (!parent.isValid()) return false; // check if the format is supported QStringList types = mimeTypes(); if (types.isEmpty()) return false; QString format = types.at(0); if (!data->hasFormat(format)) return false; QModelIndex idx = parent; // if (row >= rowCount(parent) || row < 0) { // return false; // } else { // idx= index(row,column,parent); // } ProjectModelNode* p= static_cast<ProjectModelNode*>(idx.internalPointer()); PProjectModelNode node = mProject->pointerToNode(p); if (node->isUnit) return false; QByteArray encoded = data->data(format); QDataStream stream(&encoded, QIODevice::ReadOnly); while (!stream.atEnd()) { qint32 r, c; quintptr v; stream >> r >> c >> v; ProjectModelNode* droppedPointer= (ProjectModelNode*)(v); PProjectModelNode droppedNode = mProject->pointerToNode(droppedPointer); PProjectModelNode oldParent = droppedNode->parent.lock(); if (oldParent == node) return false; } return true; } Qt::DropActions ProjectModel::supportedDropActions() const { return Qt::MoveAction; } bool ProjectModel::dropMimeData(const QMimeData *data, Qt::DropAction action, int /*row*/, int /*column*/, const QModelIndex &parent) { // check if the action is supported if (!data || action != Qt::MoveAction) return false; // check if the format is supported QStringList types = mimeTypes(); if (types.isEmpty()) return false; QString format = types.at(0); if (!data->hasFormat(format)) return false; if (!parent.isValid()) return false; ProjectModelNode* p= static_cast<ProjectModelNode*>(parent.internalPointer()); PProjectModelNode node = mProject->pointerToNode(p); QByteArray encoded = data->data(format); QDataStream stream(&encoded, QIODevice::ReadOnly); QVector<int> rows,cols; QVector<intptr_t> pointers; while (!stream.atEnd()) { qint32 r, c; quintptr v; stream >> r >> c >> v; rows.append(r); cols.append(c); pointers.append(v); } for (int i=pointers.count()-1;i>=0;i--) { int r = rows[i]; intptr_t v = pointers[i]; ProjectModelNode* droppedPointer= (ProjectModelNode*)(v); PProjectModelNode droppedNode = mProject->pointerToNode(droppedPointer); PProjectModelNode oldParent = droppedNode->parent.lock(); if (oldParent) { QModelIndex oldParentIndex = getNodeIndex(oldParent.get()); beginRemoveRows(oldParentIndex,r,r); oldParent->children.removeAt(r); endRemoveRows(); } QModelIndex newParentIndex = getNodeIndex(node.get()); beginInsertRows(newParentIndex,node->children.count(),node->children.count()); droppedNode->parent = node; node->children.append(droppedNode); if (droppedNode->isUnit) { PProjectUnit unit = droppedNode->pUnit.lock(); unit->setFolder(mProject->getNodePath(node)); } endInsertRows(); mProject->saveAll(); return true; } return false; } QMimeData *ProjectModel::mimeData(const QModelIndexList &indexes) const { if (indexes.count() <= 0) return nullptr; QStringList types = mimeTypes(); if (types.isEmpty()) return nullptr; QMimeData *data = new QMimeData(); QString format = types.at(0); QByteArray encoded; QDataStream stream(&encoded, QIODevice::WriteOnly); QModelIndexList::ConstIterator it = indexes.begin(); QList<QUrl> urls; for (; it != indexes.end(); ++it) { stream << (qint32)((*it).row()) << (qint32)((*it).column()) << (quintptr)((*it).internalPointer()); ProjectModelNode* p = static_cast<ProjectModelNode*>((*it).internalPointer()); if (p && p->isUnit) { PProjectUnit unit = p->pUnit.lock(); if (unit) urls.append(QUrl::fromLocalFile(unit->fileName())); } } if (!urls.isEmpty()) data->setUrls(urls); data->setData(format, encoded); return data; } ProjectModelSortFilterProxy::ProjectModelSortFilterProxy(QObject *parent): QSortFilterProxyModel(parent) { } bool ProjectModelSortFilterProxy::lessThan(const QModelIndex &source_left, const QModelIndex &source_right) const { if (!sourceModel()) return false; ProjectModelNode* pLeft=nullptr; if (source_left.isValid()) pLeft = static_cast<ProjectModelNode*>(source_left.internalPointer()); ProjectModelNode* pRight=nullptr; if (source_right.isValid()) pRight = static_cast<ProjectModelNode*>(source_right.internalPointer()); if (!pLeft) return true; if (!pRight) return false; if (!pLeft->isUnit && pRight->isUnit) return true; if (pLeft->isUnit && !pRight->isUnit) return false; if (pLeft->priority!=pRight->priority) return pLeft->priority>pRight->priority; return QString::compare(pLeft->text, pRight->text)<0; }