/* # --------------------------------------------------------------------- # # Copyright (c) CREATIS (Centre de Recherche en Acquisition et Traitement de l'Image # pour la Sant�) # Authors : Eduardo Davila, Frederic Cervenansky, Claire Mouton # Previous Authors : Laurent Guigues, Jean-Pierre Roux # CreaTools website : www.creatis.insa-lyon.fr/site/fr/creatools_accueil # # This software is governed by the CeCILL-B license under French law and # abiding by the rules of distribution of free software. You can use, # modify and/ or redistribute the software under the terms of the CeCILL-B # license as circulated by CEA, CNRS and INRIA at the following URL # http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.html # or in the file LICENSE.txt. # # As a counterpart to the access to the source code and rights to copy, # modify and redistribute granted by the license, users are provided only # with a limited warranty and the software's author, the holder of the # economic rights, and the successive licensors have only limited # liability. # # The fact that you are presently reading this means that you have had # knowledge of the CeCILL-B license and that you accept its terms. # ------------------------------------------------------------------------ */ /* * modelCDMAppli.cpp * * Created on: Nov 23, 2012 * Author: Daniel Felipe Gonzalez Obando */ #include "modelCDMAppli.h" #include #include #include #include #include "CDMUtilities.h" #include "creaWx.h" #include "wx/dir.h" modelCDMAppli::modelCDMAppli() { } modelCDMAppli::modelCDMAppli(modelCDMIProjectTreeNode* parent, const std::string& path, const std::string& name, const int& level) { std::cout << "creating appli\n"; this->parent = parent; this->type = wxDIR_DIRS; this->name = name; this->level = level; this->path = path; this->path = CDMUtilities::fixPath(path); std::string pathFixed(CDMUtilities::fixPath(path)); this->applications.clear(); wxDir dir(crea::std2wx((pathFixed).c_str())); if (dir.IsOpened()) { wxString fileName; bool cont = dir.GetFirst(&fileName, wxEmptyString, wxDIR_DIRS); while (cont) { std::string stdfileName = crea::wx2std(fileName); if(stdfileName != "template_appli" && stdfileName != "template_wx_appli") { modelCDMApplication* application = new modelCDMApplication(this, pathFixed + CDMUtilities::SLASH + stdfileName, stdfileName, this->level + 1); this->applications.push_back(application); this->children.push_back(application); } else { modelCDMFolder* folder = new modelCDMFolder(this, pathFixed + CDMUtilities::SLASH + stdfileName, stdfileName, this->level + 1); this->children.push_back(folder); } cont = dir.GetNext(&fileName); } //files cont = dir.GetFirst(&fileName, wxEmptyString, wxDIR_FILES); while (cont) { std::string stdfileName = crea::wx2std(fileName); //if CMakeLists, create CMakeLists if(stdfileName == "CMakeLists.txt") { this->CMakeLists = new modelCDMCMakeListsFile(this, pathFixed + CDMUtilities::SLASH + stdfileName, stdfileName, this->level + 1); this->children.push_back(this->CMakeLists); } //if is an unknown file, create file else { this->children.push_back(new modelCDMFile(this, pathFixed + CDMUtilities::SLASH + stdfileName, stdfileName, this->level + 1)); } cont = dir.GetNext(&fileName); } } this->SortChildren(); std::sort(this->applications.begin(), this->applications.end(), CompareNodeItem); } modelCDMAppli::~modelCDMAppli() { } const std::vector& modelCDMAppli::GetApplications() const { return this->applications; } modelCDMApplication* modelCDMAppli::CreateApplication( const std::string& namein, const int& type, std::string*& result ) { std::vector words; CDMUtilities::splitter::split(words,namein," '/\\*\"%",CDMUtilities::splitter::no_empties); std::string name; for (int i = 0; i < (int)(words.size()); i++) { name += words[i]; } if (name == "") { result = new std::string("The given name is not valid: '/\\*\"% are forbidden."); return NULL; } if (type == 0) { //copy template application folder with new name #ifdef _WIN32 std::string copyCommand = "xcopy \"" + this->path + CDMUtilities::SLASH + "template_appli\" \"" + this->path + CDMUtilities::SLASH + name + CDMUtilities::SLASH + "\" /Y"; #else std::string copyCommand = "cp -r \"" + this->path + CDMUtilities::SLASH + "template_appli\" \"" + this->path + CDMUtilities::SLASH + name + "\""; #endif if(system(copyCommand.c_str())) { result = new std::string("An error occurred while running '" + copyCommand + "'."); return NULL; } //set name of library in CMakeLists inside copied folder std::string line; std::ifstream in((this->path + CDMUtilities::SLASH + name + CDMUtilities::SLASH + "CMakeLists.txt").c_str()); if( !in.is_open()) { result = new std::string("CMakeLists.txt file failed to open."); return NULL; } std::ofstream out((this->path + CDMUtilities::SLASH + name + CDMUtilities::SLASH + "CMakeLists.txt.tmp").c_str()); while (getline(in, line)) { if(line == "SET ( EXE_NAME MyExe )") line = "SET ( EXE_NAME " + name + " )"; out << line << std::endl; } in.close(); out.close(); //delete old file and rename new file #ifdef _WIN32 std::string renameCommand = "move /Y \"" + this->path + CDMUtilities::SLASH + name + CDMUtilities::SLASH + "CMakeLists.txt.tmp\" \"" + this->path + CDMUtilities::SLASH + name + CDMUtilities::SLASH + "CMakeLists.txt\""; #else std::string renameCommand = "mv \"" + this->path + CDMUtilities::SLASH + name + CDMUtilities::SLASH + "CMakeLists.txt.tmp\" \"" + this->path + CDMUtilities::SLASH + name + CDMUtilities::SLASH + "CMakeLists.txt\""; #endif if(system(renameCommand.c_str())) { result = new std::string("An error occurred while running '" + renameCommand + "'."); return NULL; } //add application to appli CMakeLists std::fstream out1((this->path + CDMUtilities::SLASH + "CMakeLists.txt").c_str(), std::fstream::in | std::fstream::out | std::fstream::app); if (out1.is_open()) { out1 << "ADD_SUBDIRECTORY(" << name << ")" << std::endl; out1.close(); } //add application to model modelCDMApplication* application = new modelCDMApplication(this, this->path + CDMUtilities::SLASH + name, name, this->level + 1); this->applications.push_back(application); this->children.push_back(application); this->SortChildren(); result = new std::string(this->path + CDMUtilities::SLASH + name); return application; } else if(type == 1) { //copy template application folder with new name #ifdef _WIN32 std::string copyCommand = "xcopy \"" + this->path + CDMUtilities::SLASH + "template_wx_appli\" \"" + this->path + CDMUtilities::SLASH + name + CDMUtilities::SLASH + "\" /Y"; #else std::string copyCommand = "cp -r \"" + this->path + CDMUtilities::SLASH + "template_wx_appli\" \"" + this->path + CDMUtilities::SLASH + name + "\""; #endif if(system(copyCommand.c_str())) { result = new std::string("An error occurred while running '" + copyCommand + "'."); return NULL; } //set name of library in CMakeLists inside copied folder std::string line; std::ifstream in((this->path + CDMUtilities::SLASH + name + CDMUtilities::SLASH + "CMakeLists.txt").c_str()); if( !in.is_open()) { result = new std::string("CMakeLists.txt file failed to open."); return NULL; } std::ofstream out((this->path + CDMUtilities::SLASH + name + CDMUtilities::SLASH + "CMakeLists.txt.tmp").c_str()); while (getline(in, line)) { if(line == "SET ( EXE_NAME MyExeWx )") line = "SET ( EXE_NAME " + name + " )"; out << line << std::endl; } in.close(); out.close(); //delete old file and rename new file #ifdef _WIN32 std::string renameCommand = "move /Y \"" + this->path + CDMUtilities::SLASH + name + CDMUtilities::SLASH + "CMakeLists.txt.tmp\" \"" + this->path + CDMUtilities::SLASH + name + CDMUtilities::SLASH + "CMakeLists.txt\""; #else std::string renameCommand = "mv \"" + this->path + CDMUtilities::SLASH + name + CDMUtilities::SLASH + "CMakeLists.txt.tmp\" \"" + this->path + CDMUtilities::SLASH + name + CDMUtilities::SLASH + "CMakeLists.txt\""; #endif if(system(renameCommand.c_str())) { result = new std::string("An error occurred while running '" + renameCommand + "'."); return NULL; } //add application to appli CMakeLists std::fstream out1((this->path + CDMUtilities::SLASH + "CMakeLists.txt").c_str(), std::fstream::in | std::fstream::out | std::fstream::app); if (out1.is_open()) { out1 << "ADD_SUBDIRECTORY(" << name << ")" << std::endl; out1.close(); } //add application to model modelCDMApplication* application = new modelCDMApplication(this, this->path + CDMUtilities::SLASH + name, name, this->level + 1); this->applications.push_back(application); this->children.push_back(application); this->SortChildren(); result = new std::string(this->path + CDMUtilities::SLASH + name); return application; } else { std::string res = "Invalid application type: "; res += type; res += std::string(".\n0:Console application.\n1:GUI Application (wxWidgets)."); result = new std::string(res); return NULL; } } const bool modelCDMAppli::Refresh(std::string*& result) { std::cout << "refreshing appli" << std::endl; this->type = wxDIR_DIRS; std::vector checked(this->children.size(), false); std::vector checkedApplications(this->applications.size(), false); //check all folders wxDir dir(crea::std2wx((this->path).c_str())); if (dir.IsOpened()) { wxString fileName; bool cont = dir.GetFirst(&fileName, wxEmptyString, wxDIR_DIRS); while (cont) { std::string stdfileName = crea::wx2std(fileName); if(stdfileName != "template_appli" && stdfileName != "template_wx_appli") { std::string applicationName = stdfileName; //check if application already exist bool found = false; for (int i = 0; !found && i < (int)(this->applications.size()); i++) { if (this->applications[i]->GetName() == applicationName) { found = true; int pos = std::find(this->children.begin(), this->children.end(), this->applications[i]) - this->children.begin(); checked[pos] = true; checkedApplications[i] = true; if(!this->applications[i]->Refresh(result)) return false; } } if(!found) { modelCDMApplication* application= new modelCDMApplication(this, this->path + CDMUtilities::SLASH + stdfileName, stdfileName, this->level + 1); this->applications.push_back(application); this->children.push_back(application); } } else { //check if folder already exist bool found = false; for (int i = 0; !found && i < (int)(this->children.size()); i++) { if (this->children[i]->GetName() == stdfileName) { found = true; checked[i] = true; if(!this->children[i]->Refresh(result)) return false; } } if(!found) { modelCDMFolder* folder= new modelCDMFolder(this, this->path + CDMUtilities::SLASH + stdfileName, stdfileName, this->level + 1); this->children.push_back(folder); } } cont = dir.GetNext(&fileName); } cont = dir.GetFirst(&fileName, wxEmptyString, wxDIR_FILES); while (cont) { std::string stdfileName = crea::wx2std(fileName); //if CMakeLists, create CMakeLists if(stdfileName == "CMakeLists.txt") { if (this->CMakeLists == NULL) { this->CMakeLists = new modelCDMCMakeListsFile(this, this->path + CDMUtilities::SLASH + stdfileName, stdfileName, this->level + 1); this->children.push_back(this->CMakeLists); } else { int pos = std::find(this->children.begin(), this->children.end(), this->CMakeLists) - this->children.begin(); checked[pos] = true; if(!this->CMakeLists->Refresh(result)) return false; } } //if is an unknown file, create file else { bool found = false; for (int i = 0; !found && i < (int)(this->children.size()); i++) { if (this->children[i]->GetName() == stdfileName) { found = true; checked[i] = true; if(!this->children[i]->Refresh(result)) return false; } } if(!found) { modelCDMFile* file = new modelCDMFile(this, this->path + CDMUtilities::SLASH + stdfileName, stdfileName, this->level + 1); this->children.push_back(file); } } cont = dir.GetNext(&fileName); } } for (int i = 0; i < (int)(checkedApplications.size()); i++) { if(!checkedApplications[i]) { this->applications.erase(this->applications.begin()+i); checkedApplications.erase(checkedApplications.begin()+i); i--; } } for (int i = 0; i < (int)(checked.size()); i++) { if(!checked[i]) { delete this->children[i]; this->children.erase(this->children.begin()+i); checked.erase(checked.begin()+i); i--; } } this->SortChildren(); std::sort(this->applications.begin(), this->applications.end(), CompareNodeItem); return true; } void modelCDMAppli::CheckStructure(std::map& properties) { //check cmake exist if(this->CMakeLists != NULL) { //set properties parameters based on model for (int i = 0; i < (int)(this->applications.size()); i++) properties["appli add " + this->applications[i]->GetName()] = false; //open cmakelists std::ifstream confFile; confFile.open((this->CMakeLists->GetPath()).c_str()); //take everything that is not commented std::string fileContent; std::string word; std::vector words; while(confFile.is_open() && !confFile.eof()) { std::getline(confFile,word, '\n'); if(word[0] != '#') { CDMUtilities::splitter::split(words, word, "#", CDMUtilities::splitter::empties_ok); if (words.size() > 0) { word = words[0]; CDMUtilities::splitter::split(words, word, " ", CDMUtilities::splitter::empties_ok); for (int i = 0; i < (int)(words.size()); i++) { if(words[i].substr(0,2) == "//") break; fileContent += words[i] + " "; } } } } //check every instruction std::stringstream ss(fileContent); while(!ss.eof()) { std::getline(ss,word, '('); //check instruction name CDMUtilities::splitter::split(words, word, " ", CDMUtilities::splitter::no_empties); //add instructions if (words.size() > 0 && words[words.size()-1] == "ADD_SUBDIRECTORY") { std::getline(ss,word, ')'); //std::cout << word << std::endl; CDMUtilities::splitter::split(words, word, " ", CDMUtilities::splitter::no_empties); if (words.size() > 0) { { properties["appli add " + words[0]] = true; } } } } } //check libraries' structure for (int i = 0; i < (int)(this->applications.size()); i++) { properties["application " + this->applications[i]->GetName()] = true; this->applications[i]->CheckStructure(properties); } } bool modelCDMAppli::IsApplicationIncluded(const std::string& application_name) { if(this->HasCMakeLists()) { std::ifstream CMFile(this->CMakeLists->GetPath().c_str()); if (CMFile.is_open()) { std::string line; while(!CMFile.eof()) { std::getline(CMFile, line); while(line[0]==' ') line.erase(0); if(line[0] != '#') { std::vector lineSeg; CDMUtilities::splitter::split(lineSeg,line,"()",CDMUtilities::splitter::no_empties); if(lineSeg.size() > 0 && lineSeg[0] == "ADD_SUBDIRECTORY" && lineSeg[1] == application_name) { CMFile.close(); return true; } } } CMFile.close(); } } return false; } bool modelCDMAppli::SetApplicationInclude(const std::string& application_name, const bool& toInclude) { if (this->HasCMakeLists()) { std::ifstream CMFile(this->CMakeLists->GetPath().c_str()); if (CMFile.is_open()) { std::stringstream outs; std::string line; bool found = false; while(!CMFile.eof()) { std::getline(CMFile, line); if (CMFile.eof()) { break; } if(line != "") { std::vector segs; CDMUtilities::splitter::split(segs, line, " ", CDMUtilities::splitter::no_empties); //is comment if(segs.size() > 0 && segs[0][0] == '#') { if(toInclude) { CDMUtilities::splitter::split(segs, line, " #()", CDMUtilities::splitter::no_empties); if (segs.size() > 1 && segs[0] == "ADD_SUBDIRECTORY" && segs[1] == application_name) { found = true; outs << "ADD_SUBDIRECTORY(" << application_name << ")\n"; } else outs << line << "\n"; } else { outs << line << "\n"; } } //is not comment else { if (segs.size() > 0 && !toInclude) { CDMUtilities::splitter::split(segs, line, " ()", CDMUtilities::splitter::no_empties); if (segs.size() > 1 && segs[0] == "ADD_SUBDIRECTORY" && segs[1] == application_name) { outs << "#" << line << "\n"; } else { outs << line << "\n"; } } else { CDMUtilities::splitter::split(segs, line, " ()", CDMUtilities::splitter::no_empties); if (segs.size() > 1 && segs[0] == "ADD_SUBDIRECTORY" && segs[1] == application_name) { found = true; } outs << line << "\n"; } } } else { outs << "\n"; } } CMFile.close(); if(!found && toInclude) outs << "ADD_SUBDIRECTORY(" << application_name << ")\n"; std::ofstream CMFileOut(this->CMakeLists->GetPath().c_str()); if (CMFileOut.is_open()) { CMFileOut << outs.rdbuf(); CMFileOut.close(); return true; } } } return false; }