/*========================================================================= Program: CMake - Cross-Platform Makefile Generator Module: $RCSfile$ Language: C++ Date: $Date$ Version: $Revision$ Copyright (c) 2002 Kitware, Inc., Insight Consortium. All rights reserved. Copyright (c) 2004 Alexander Neundorf neundorf@kde.org, All rights reserved. See Copyright.txt or http://www.cmake.org/HTML/Copyright.html for details. This software is distributed WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the above copyright notices for more information. =========================================================================*/ #include "cmGlobalKdevelopGenerator.h" #include "cmLocalKdevelopGenerator.h" #include "cmMakefile.h" #include "cmake.h" #include "cmSourceFile.h" #include "cmGeneratedFileStream.h" cmGlobalKdevelopGenerator::cmGlobalKdevelopGenerator() { // This type of makefile always requires unix style paths this->ForceUnixPaths = true; this->FindMakeProgramFile = "CMakeUnixFindMake.cmake"; this->ToolSupportsColor = false; } ///! Create a local generator appropriate to this Global Generator cmLocalGenerator *cmGlobalKdevelopGenerator::CreateLocalGenerator() { cmLocalGenerator *lg = new cmLocalKdevelopGenerator; lg->SetGlobalGenerator(this); return lg; } //---------------------------------------------------------------------------- void cmGlobalKdevelopGenerator ::GetDocumentation(cmDocumentationEntry& entry) const { entry.name = this->GetName(); entry.brief = "Generates KDevelop 3 project files."; entry.full = "Project files for KDevelop 3 will be created in the top directory " "and in every subdirectory which features a CMakeLists.txt file " "containing a PROJECT() call. " "If you change the settings using KDevelop cmake will try its best " "to keep your changes when regenerating the project files. " "Additionally a hierarchy of UNIX makefiles is generated into the " "build tree. Any " "standard UNIX-style make program can build the project through the " "default make target. A \"make install\" target is also provided."; } void cmGlobalKdevelopGenerator::Generate() { this->cmGlobalUnixMakefileGenerator3::Generate(); // for each sub project in the project create // a kdevelop project std::map >::iterator it; for(it = this->ProjectMap.begin(); it!= this->ProjectMap.end(); ++it) { cmMakefile* mf = it->second[0]->GetMakefile(); std::string outputDir=mf->GetStartOutputDirectory(); std::string projectDir=mf->GetHomeDirectory(); std::string projectName=mf->GetProjectName(); std::string cmakeFilePattern("CMakeLists.txt;*.cmake;"); std::string fileToOpen; std::vector& lgs= it->second; // create the project.kdevelop.filelist file if(!this->CreateFilelistFile(it->second[0], lgs, outputDir, projectDir, projectName, cmakeFilePattern, fileToOpen)) { cmSystemTools::Error("Can not create filelist file"); return; } //try to find the name of an executable so we have something to //run from kdevelop for now just pick the first executable found std::string executable; for (std::vector::const_iterator lg=lgs.begin(); lg!=lgs.end(); lg++) { cmMakefile* makefile=(*lg)->GetMakefile(); cmTargets& targets=makefile->GetTargets(); for (cmTargets::iterator ti = targets.begin(); ti != targets.end(); ti++) { if (ti->second.GetType()==cmTarget::EXECUTABLE) { executable = ti->second.GetProperty("LOCATION"); break; } } if (!executable.empty()) { break; } } // now create a project file this->CreateProjectFile(outputDir, projectDir, projectName, executable, cmakeFilePattern, fileToOpen); } } bool cmGlobalKdevelopGenerator ::CreateFilelistFile(cmLocalGenerator* , std::vector& lgs, const std::string& outputDir, const std::string& projectDirIn, const std::string& projectname, std::string& cmakeFilePattern, std::string& fileToOpen) { std::string projectDir = projectDirIn + "/"; std::string filename = outputDir+ "/" + projectname +".kdevelop.filelist"; std::set files; std::string tmp; for (std::vector::const_iterator it=lgs.begin(); it!=lgs.end(); it++) { cmMakefile* makefile=(*it)->GetMakefile(); const std::vector& listFiles=makefile->GetListFiles(); for (std::vector::const_iterator lt=listFiles.begin(); lt!=listFiles.end(); lt++) { tmp=*lt; cmSystemTools::ReplaceString(tmp, projectDir.c_str(), ""); // make sure the file is part of this source tree if ((tmp[0]!='/') && (strstr(tmp.c_str(), "CMakeFiles/")==0)) { files.insert(tmp); tmp=cmSystemTools::GetFilenameName(tmp); //add all files which dont match the default // */CMakeLists.txt;*cmake; to the file pattern if ((tmp!="CMakeLists.txt") && (strstr(tmp.c_str(), ".cmake")==0)) { cmakeFilePattern+=tmp+";"; } } } //get all sources cmTargets& targets=makefile->GetTargets(); for (cmTargets::iterator ti = targets.begin(); ti != targets.end(); ti++) { const std::vector& sources=ti->second.GetSourceFiles(); for (std::vector::const_iterator si=sources.begin(); si!=sources.end(); si++) { tmp=(*si)->GetFullPath(); cmSystemTools::ReplaceString(tmp, projectDir.c_str(), ""); if ((tmp[0]!='/') && (strstr(tmp.c_str(), "CMakeFiles/")==0)) { files.insert(tmp); } } for (std::vector::const_iterator lt=listFiles.begin(); lt!=listFiles.end(); lt++) { tmp=*lt; cmSystemTools::ReplaceString(tmp, projectDir.c_str(), ""); if ((tmp[0]!='/') && (strstr(tmp.c_str(), "CMakeFiles/")==0)) { files.insert(tmp.c_str()); } } } } //check if the output file already exists and read it //insert all files which exist into the set of files std::ifstream oldFilelist(filename.c_str()); if (oldFilelist) { while (cmSystemTools::GetLineFromStream(oldFilelist, tmp)) { if (tmp[0]=='/') { continue; } std::string completePath=projectDir+tmp; if (cmSystemTools::FileExists(completePath.c_str())) { files.insert(tmp); } } oldFilelist.close(); } //now write the new filename cmGeneratedFileStream fout(filename.c_str()); if(!fout) { return false; } fileToOpen=""; for (std::set::const_iterator it=files.begin(); it!=files.end(); it++) { // get the full path to the file tmp=cmSystemTools::CollapseFullPath(it->c_str(), projectDir.c_str()); // just select the first source file if (fileToOpen.empty()) { std::string ext = cmSystemTools::GetFilenameExtension(tmp); if ((ext==".c") || (ext==".cc") || (ext==".cpp") || (ext==".C") || (ext==".h")) { fileToOpen=tmp; } } // make it relative to the project dir cmSystemTools::ReplaceString(tmp, projectDir.c_str(), ""); // only put relative paths if (tmp.size() && tmp[0] != '/') { fout << tmp.c_str() <<"\n"; } } return true; } /* create the project file, if it already exists, merge it with the existing one, otherwise create a new one */ void cmGlobalKdevelopGenerator ::CreateProjectFile(const std::string& outputDir, const std::string& projectDir, const std::string& projectname, const std::string& executable, const std::string& cmakeFilePattern, const std::string& fileToOpen) { std::string filename=outputDir+"/"; filename+=projectname+".kdevelop"; std::string sessionFilename=outputDir+"/"; sessionFilename+=projectname+".kdevses"; if (cmSystemTools::FileExists(filename.c_str())) { this->MergeProjectFiles(outputDir, projectDir, filename, executable, cmakeFilePattern, fileToOpen, sessionFilename); } else { this->CreateNewProjectFile(outputDir, projectDir, filename, executable, cmakeFilePattern, fileToOpen, sessionFilename); } } void cmGlobalKdevelopGenerator ::MergeProjectFiles(const std::string& outputDir, const std::string& projectDir, const std::string& filename, const std::string& executable, const std::string& cmakeFilePattern, const std::string& fileToOpen, const std::string& sessionFilename) { std::ifstream oldProjectFile(filename.c_str()); if (!oldProjectFile) { this->CreateNewProjectFile(outputDir, projectDir, filename, executable, cmakeFilePattern, fileToOpen, sessionFilename); return; } /* Read the existing project file (line by line), copy all lines into the new project file, except the ones which can be reliably set from contents of the CMakeLists.txt */ std::string tmp; std::vector lines; while (cmSystemTools::GetLineFromStream(oldProjectFile, tmp)) { lines.push_back(tmp); } oldProjectFile.close(); cmGeneratedFileStream fout(filename.c_str()); if(!fout) { return; } for (std::vector::const_iterator it=lines.begin(); it!=lines.end(); it++) { const char* line=(*it).c_str(); // skip these tags as they are always replaced if ((strstr(line, "")!=0) || (strstr(line, "")!=0) || (strstr(line, "")!=0) || (strstr(line, "")!=0) || (strstr(line, "")!=0) || (strstr(line, "")!=0)) { continue; } // output the line from the file if it is not one of the above tags fout<<*it<<"\n"; // if this is the tag output the stuff that goes in the // general tag if (strstr(line, "")) { fout<< " KDevCustomProject\n"; fout<< " " <\n"; //this one is important fout<<" true\n"; //and this one } // inside kdevcustomproject the must be put if (strstr(line, "")) { fout<<" "<\n"; } // buildtool and builddir go inside if (strstr(line, "")) { fout<<" make\n"; fout<<" "<\n"; } } } void cmGlobalKdevelopGenerator ::CreateNewProjectFile(const std::string& outputDir, const std::string& projectDir, const std::string& filename, const std::string& executable, const std::string& cmakeFilePattern, const std::string& fileToOpen, const std::string& sessionFilename) { cmGeneratedFileStream fout(filename.c_str()); if(!fout) { return; } fout<<"\n"; fout<<"\n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" $VERSION$\n"; fout<<" KDevCustomProject\n"; fout<<" C++\n"; fout<<" \n"; fout<<" "<\n"; //this one is important fout<<" true\n"; //and this one fout<<" \n"; fout<<" C\n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" "<\n"; fout<<" \n"; fout<<" "<\n"; fout<<" custom\n"; fout<<" /\n"; fout<<" \n"; fout<<" false\n"; fout<<" true\n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" make\n"; //this one is important fout<<" "<\n"; //and this one fout<<" \n"; fout<<" \n"; fout<<" false\n"; fout<<" 1\n"; fout<<" false\n"; fout<<" \n"; fout<<" default\n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" html/\n"; fout<<" html/\n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" .h\n"; fout<<" .cpp\n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" true\n"; fout<<" true\n"; fout<<" true\n"; fout<<" false\n"; fout<<" true\n"; fout<<" true\n"; fout<<" true\n"; fout<<" 250\n"; fout<<" 400\n"; fout<<" 250\n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" \n"; fout<<" true\n"; fout<<" \n"; fout<<" \n"; fout<<" *.o,*.lo,CVS,*~,cmake*\n"; fout<<" true\n"; fout<<" \n"; fout<<" \n"; fout<<"\n"; if (sessionFilename.empty()) return; // and a session file, so that kdevelop opens a file if it opens the // project the first time cmGeneratedFileStream devses(sessionFilename.c_str()); if(!devses) { return; } devses<<"\n"; devses<<"\n"; devses<<"\n"; devses<<" \n"; devses<<" \n"; devses<<" \n"; devses<<" \n"; devses<<" \n"; devses<<"\n"; }