1 // Copyright (C) 2014-2023 CEA/DEN, EDF R&D
3 // This library is free software; you can redistribute it and/or
4 // modify it under the terms of the GNU Lesser General Public
5 // License as published by the Free Software Foundation; either
6 // version 2.1 of the License, or (at your option) any later version.
8 // This library is distributed in the hope that it will be useful,
9 // but WITHOUT ANY WARRANTY; without even the implied warranty of
10 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
11 // Lesser General Public License for more details.
13 // You should have received a copy of the GNU Lesser General Public
14 // License along with this library; if not, write to the Free Software
15 // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17 // See http://www.salome-platform.org/ or email : webmaster.salome@opencascade.com
20 #include "SHAPERGUI_DataModel.h"
21 #include "SHAPERGUI.h"
23 #include <XGUI_Workshop.h>
25 #include <ModelAPI_Session.h>
26 #include <ModelAPI_AttributeString.h>
27 #include <ModelAPI_AttributeBoolean.h>
28 #include <ExchangePlugin_Dump.h>
30 #include <LightApp_Study.h>
31 #include <CAM_Application.h>
32 #include <CAM_DataObject.h>
33 #include <SUIT_Tools.h>
34 #include <SUIT_ResourceMgr.h>
38 #include <QTextStream>
40 #define DUMP_NAME "shaper_dump.py"
43 SHAPERGUI_DataModel::SHAPERGUI_DataModel(SHAPERGUI* theModule)
44 : LightApp_DataModel(theModule), myStudyPath(""), myModule(theModule)
48 SHAPERGUI_DataModel::~SHAPERGUI_DataModel()
52 bool SHAPERGUI_DataModel::open(const QString& thePath, CAM_Study* theStudy, QStringList theFiles)
54 LightApp_DataModel::open( thePath, theStudy, theFiles );
55 if (theFiles.size() == 0)
58 myStudyPath = thePath;
60 // If the file is Multi(contain all module files inside), the open SALOME functionality creates
61 // these files in a temporary directory. After the open functionality is finished, it removes
62 // these files (in the full SALOME mode).
63 // The postponed loading of the files is realized in the SHAPER module. So, it is important do
64 // not remove the opened files.
65 // The following code creates a new tmp directory with a copy of files.
66 QString aTmpDir = theFiles.first();
68 LightApp_Study* aStudy = dynamic_cast<LightApp_Study*>( myModule->application()->activeStudy() );
69 QString aNewTmpDir = aStudy->GetTmpDir("", false).c_str();
72 QStringList aFiles = aDir.entryList(QDir::Files);
73 QStringList::const_iterator anIt = aFiles.begin(), aLast = aFiles.end();
74 for (; anIt != aLast; anIt++) {
75 QString aFileName = *anIt;
77 QString aCurrentFile = SUIT_Tools::addSlash(aTmpDir) + aFileName;
78 XGUI_Workshop* aWorkShop = myModule->workshop();
79 aWorkShop->openFile(aCurrentFile);
82 myModule->setIsOpened(true);
86 bool SHAPERGUI_DataModel::save(QStringList& theFiles)
88 // Publish to study before saving of the data model
89 myModule->publishToStudy();
91 LightApp_DataModel::save( theFiles );
92 XGUI_Workshop* aWorkShop = myModule->workshop();
93 std::list<std::string> aFileNames;
95 CAM_Application* anApp = myModule->application();
96 LightApp_Study* aStudy = dynamic_cast<LightApp_Study*>(anApp->activeStudy());
97 SUIT_ResourceMgr* aResMgr = anApp->resourceMgr();
99 // it is important to check whether the file is saved in the multi-files mode in order to save
100 // files in temporary directories, which are removed in the full SALOME mode after copiying
101 // the files content in a result file.
102 bool isMultiFile = aResMgr ? aResMgr->booleanValue("Study", "multi_file", false) : false;
104 std::string aTmpDir = aStudy->GetTmpDir(qPrintable(myStudyPath), isMultiFile);
105 QString aTmp = QString(aTmpDir.c_str());
106 theFiles.append(aTmp);
108 SessionPtr aMgr = ModelAPI_Session::get();
109 if (aMgr->isAutoUpdateBlocked())
110 aMgr->blockAutoUpdate(false);
112 //aWorkShop->saveDocument(QString(aTmpDir.c_str()), aFileNames);
113 aWorkShop->setCurrentDataFile(aTmp + "shaper.shaper");
115 QString aName = aWorkShop->currentDataFile();
116 aName.replace(QChar('\\'), QChar('/'));
117 int aN = aName.lastIndexOf('/');
118 theFiles.append(aName.right(aName.length() - aN - 1));
123 bool SHAPERGUI_DataModel::saveAs(const QString& thePath, CAM_Study* theStudy, QStringList& theFiles)
125 myStudyPath = thePath;
126 return save(theFiles);
129 bool SHAPERGUI_DataModel::close()
131 myModule->workshop()->closeDocument();
132 return LightApp_DataModel::close();
135 bool SHAPERGUI_DataModel::create(CAM_Study* theStudy)
140 bool SHAPERGUI_DataModel::isModified() const
142 SessionPtr aMgr = ModelAPI_Session::get();
143 return aMgr->isModified();
146 bool SHAPERGUI_DataModel::isSaved() const
148 return !isModified();
151 void SHAPERGUI_DataModel::update(LightApp_DataObject* theObj, LightApp_Study* theStudy)
153 // Nothing to do here: we always keep the data tree in the up-to-date state
154 // The only goal of this method is to hide default behavior from LightApp_DataModel
158 void SHAPERGUI_DataModel::initRootObject()
160 LightApp_Study* study = dynamic_cast<LightApp_Study*>( module()->application()->activeStudy() );
161 CAM_ModuleObject *aModelRoot = dynamic_cast<CAM_ModuleObject*>(root());
162 if(study && aModelRoot == NULL) {
163 aModelRoot = createModuleObject( study->root() );
164 aModelRoot->setDataModel( this );
169 void SHAPERGUI_DataModel::removeDirectory(const QString& theDirectoryName)
171 Qtx::rmDir(theDirectoryName);
174 bool SHAPERGUI_DataModel::dumpPython(const QString& thePath, CAM_Study* theStudy,
175 bool isMultiFile, QStringList& theListOfFiles)
177 LightApp_Study* aStudy = dynamic_cast<LightApp_Study*>(theStudy);
181 myModule->publishToStudy();
183 std::shared_ptr<ModelAPI_Document> aDoc = ModelAPI_Session::get()->activeDocument();
184 ModelAPI_Session::get()->startOperation(ExchangePlugin_Dump::ID());
185 FeaturePtr aFeature = aDoc->addFeature(ExchangePlugin_Dump::ID());
186 if (aFeature.get()) {
187 // keep path to the true dumping directory for external files dumping
188 AttributeStringPtr aAttr = aFeature->string(ExchangePlugin_Dump::DUMP_DIR_ID());
190 QString aDirPath = QFileInfo(thePath).path();
191 aAttr->setValue(aDirPath.toStdString());
194 // tmp path to write the script
195 std::string aTmpDir = aStudy->GetTmpDir(thePath.toStdString().c_str(), isMultiFile);
196 std::string aFileName = aTmpDir + DUMP_NAME;
198 if (QFile::exists(aFileName.c_str())) {
199 QFile::remove(aFileName.c_str());
202 aAttr = aFeature->string(ExchangePlugin_Dump::FILE_PATH_ID());
204 aAttr->setValue(aFileName);
206 aAttr = aFeature->string(ExchangePlugin_Dump::FILE_FORMAT_ID());
208 aAttr->setValue(".py");
211 aFeature->boolean(ExchangePlugin_Dump::EXPORT_VARIABLES_ID())->setValue(true);
214 ModelAPI_Session::get()->finishOperation();
216 if (QFile::exists(aFileName.c_str())) {
217 QFile aInFile(aFileName.c_str());
218 if (!aInFile.open(QIODevice::ReadOnly | QIODevice::Text))
220 QTextStream aText(&aInFile);
221 QString aTrace(aText.readAll());
226 aBuffer.push_back(QString("def RebuildData():"));
227 QStringList aList(aTrace.split("\n"));
228 foreach(QString aStr, aList) {
229 QString s = " " + aStr;
230 aBuffer.push_back(s);
232 aTrace = aBuffer.join("\n");
235 QFile aOutFile(aFileName.c_str());
236 // binary for SALOME, issue 16931
237 if (!aOutFile.open(QIODevice::WriteOnly | QIODevice::Truncate))
240 QTextStream aOut(&aOutFile);
241 aOut << aTrace << "\n";
245 theListOfFiles.append(aTmpDir.c_str());
246 theListOfFiles.append(DUMP_NAME);