Salome HOME
Merge remote-tracking branch 'remotes/origin/occ/compounds_processing'
[modules/shaper.git] / src / SHAPERGUI / SHAPERGUI_DataModel.cpp
1 // Copyright (C) 2014-2019  CEA/DEN, EDF R&D
2 //
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.
7 //
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.
12 //
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
16 //
17 // See http://www.salome-platform.org/ or email : webmaster.salome@opencascade.com
18 //
19
20 #include "SHAPERGUI_DataModel.h"
21 #include "SHAPERGUI.h"
22
23 #include <XGUI_Workshop.h>
24
25 #include <ModelAPI_Session.h>
26 #include <ModelAPI_AttributeString.h>
27 #include <ModelAPI_AttributeBoolean.h>
28 #include <ExchangePlugin_Dump.h>
29
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>
35
36 #include <QFile>
37 #include <QDir>
38 #include <QTextStream>
39
40 #define DUMP_NAME "shaper_dump.py"
41
42
43 SHAPERGUI_DataModel::SHAPERGUI_DataModel(SHAPERGUI* theModule)
44     : LightApp_DataModel(theModule), myStudyPath(""), myModule(theModule)
45 {
46 }
47
48 SHAPERGUI_DataModel::~SHAPERGUI_DataModel()
49 {
50 }
51
52 bool SHAPERGUI_DataModel::open(const QString& thePath, CAM_Study* theStudy, QStringList theFiles)
53 {
54   LightApp_DataModel::open( thePath, theStudy, theFiles );
55   if (theFiles.size() == 0)
56     return false;
57
58   myStudyPath = thePath;
59
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();
67
68   LightApp_Study* aStudy = dynamic_cast<LightApp_Study*>( myModule->application()->activeStudy() );
69   QString aNewTmpDir = aStudy->GetTmpDir("", false).c_str();
70
71   bool isDone = true;
72   QDir aDir(aTmpDir);
73   QStringList aFiles = aDir.entryList(QDir::Files);
74   QStringList::const_iterator anIt = aFiles.begin(), aLast = aFiles.end();
75   for (; anIt != aLast; anIt++) {
76     QString aFileName = *anIt;
77
78     QString aCurrentFile = SUIT_Tools::addSlash(aTmpDir) + aFileName;
79     XGUI_Workshop* aWorkShop = myModule->workshop();
80     aWorkShop->openFile(aCurrentFile);
81   }
82
83   myModule->setIsOpened(true);
84   return true;
85 }
86
87 bool SHAPERGUI_DataModel::save(QStringList& theFiles)
88 {
89   // Publish to study before saving of the data model
90   myModule->publishToStudy();
91
92   LightApp_DataModel::save( theFiles );
93   XGUI_Workshop* aWorkShop = myModule->workshop();
94   std::list<std::string> aFileNames;
95
96   CAM_Application* anApp = myModule->application();
97   LightApp_Study* aStudy = dynamic_cast<LightApp_Study*>(anApp->activeStudy());
98   SUIT_ResourceMgr* aResMgr = anApp->resourceMgr();
99
100   // it is important to check whether the file is saved in the multi-files mode in order to save
101   // files in temporary directories, which are removed in the full SALOME mode after copiying
102   // the files content in a result file.
103   bool isMultiFile = aResMgr ? aResMgr->booleanValue("Study", "multi_file", false) : false;
104
105   std::string aTmpDir = aStudy->GetTmpDir(qPrintable(myStudyPath), isMultiFile);
106   QString aTmp = QString(aTmpDir.c_str());
107   theFiles.append(aTmp);
108
109   SessionPtr aMgr = ModelAPI_Session::get();
110   if (aMgr->isAutoUpdateBlocked())
111     aMgr->blockAutoUpdate(false);
112
113   //aWorkShop->saveDocument(QString(aTmpDir.c_str()), aFileNames);
114   aWorkShop->setCurrentDataFile(aTmp + "shaper.shaper");
115   aWorkShop->onSave();
116   QString aName = aWorkShop->currentDataFile();
117   aName.replace(QChar('\\'), QChar('/'));
118   int aN = aName.lastIndexOf('/');
119   theFiles.append(aName.right(aName.length() - aN - 1));
120
121   return true;
122 }
123
124 bool SHAPERGUI_DataModel::saveAs(const QString& thePath, CAM_Study* theStudy, QStringList& theFiles)
125 {
126   myStudyPath = thePath;
127   return save(theFiles);
128 }
129
130 bool SHAPERGUI_DataModel::close()
131 {
132   myModule->workshop()->closeDocument();
133   return LightApp_DataModel::close();
134 }
135
136 bool SHAPERGUI_DataModel::create(CAM_Study* theStudy)
137 {
138   return true;
139 }
140
141 bool SHAPERGUI_DataModel::isModified() const
142 {
143   SessionPtr aMgr = ModelAPI_Session::get();
144   return aMgr->isModified();
145 }
146
147 bool SHAPERGUI_DataModel::isSaved() const
148 {
149   return !isModified();
150 }
151
152 void SHAPERGUI_DataModel::update(LightApp_DataObject* theObj, LightApp_Study* theStudy)
153 {
154   // Nothing to do here: we always keep the data tree in the up-to-date state
155   // The only goal of this method is to hide default behavior from LightApp_DataModel
156   return;
157 }
158
159 void SHAPERGUI_DataModel::initRootObject()
160 {
161   LightApp_Study* study = dynamic_cast<LightApp_Study*>( module()->application()->activeStudy() );
162   CAM_ModuleObject *aModelRoot = dynamic_cast<CAM_ModuleObject*>(root());
163   if(study && aModelRoot == NULL) {
164     aModelRoot = createModuleObject( study->root() );
165     aModelRoot->setDataModel( this );
166     setRoot(aModelRoot);
167   }
168 }
169
170 void SHAPERGUI_DataModel::removeDirectory(const QString& theDirectoryName)
171 {
172   Qtx::rmDir(theDirectoryName);
173 }
174
175 bool SHAPERGUI_DataModel::dumpPython(const QString& thePath, CAM_Study* theStudy,
176                                      bool isMultiFile,  QStringList& theListOfFiles)
177 {
178   LightApp_Study* aStudy = dynamic_cast<LightApp_Study*>(theStudy);
179   if (!aStudy)
180     return false;
181
182   myModule->publishToStudy();
183
184   std::shared_ptr<ModelAPI_Document> aDoc = ModelAPI_Session::get()->activeDocument();
185   ModelAPI_Session::get()->startOperation(ExchangePlugin_Dump::ID());
186   FeaturePtr aFeature = aDoc->addFeature(ExchangePlugin_Dump::ID());
187   if (aFeature.get()) {
188     std::string aTmpDir = aStudy->GetTmpDir(thePath.toStdString().c_str(), isMultiFile);
189     std::string aFileName = aTmpDir + DUMP_NAME;
190
191     if (QFile::exists(aFileName.c_str())) {
192       QFile::remove(aFileName.c_str());
193     }
194
195     AttributeStringPtr aAttr = aFeature->string(ExchangePlugin_Dump::FILE_PATH_ID());
196     if (aAttr.get())
197       aAttr->setValue(aFileName);
198
199     aAttr = aFeature->string(ExchangePlugin_Dump::FILE_FORMAT_ID());
200     if (aAttr.get())
201       aAttr->setValue(".py");
202
203 #ifdef HAVE_SALOME
204     aFeature->boolean(ExchangePlugin_Dump::EXPORT_VARIABLES_ID())->setValue(true);
205 #endif
206
207     ModelAPI_Session::get()->finishOperation();
208
209     if (QFile::exists(aFileName.c_str())) {
210         QFile aInFile(aFileName.c_str());
211       if (!aInFile.open(QIODevice::ReadOnly | QIODevice::Text))
212         return false;
213       QTextStream aText(&aInFile);
214       QString aTrace(aText.readAll());
215       aInFile.close();
216
217       if (isMultiFile) {
218         QStringList aBuffer;
219         aBuffer.push_back(QString("def RebuildData():"));
220         QStringList aList(aTrace.split("\n"));
221         foreach(QString aStr, aList) {
222           QString s = "  " + aStr;
223           aBuffer.push_back(s);
224         }
225         aTrace = aBuffer.join("\n");
226       }
227
228       QFile aOutFile(aFileName.c_str());
229       // binary for SALOME, issue 16931
230       if (!aOutFile.open(QIODevice::WriteOnly | QIODevice::Truncate))
231         return false;
232
233       QTextStream aOut(&aOutFile);
234       aOut << aTrace.toStdString().c_str() << "\n";
235       aOut.flush();
236       aOutFile.close();
237
238       theListOfFiles.append(aTmpDir.c_str());
239       theListOfFiles.append(DUMP_NAME);
240       return true;
241     }
242   }
243   return false;
244 }