Salome HOME
Use the prefix std:: instead of the directive using namespace std;
[modules/geom.git] / src / GEOMToolsGUI / GEOMToolsGUI.cxx
1 //  Copyright (C) 2007-2008  CEA/DEN, EDF R&D, OPEN CASCADE
2 //
3 //  Copyright (C) 2003-2007  OPEN CASCADE, EADS/CCR, LIP6, CEA/DEN,
4 //  CEDRAT, EDF R&D, LEG, PRINCIPIA R&D, BUREAU VERITAS
5 //
6 //  This library is free software; you can redistribute it and/or
7 //  modify it under the terms of the GNU Lesser General Public
8 //  License as published by the Free Software Foundation; either
9 //  version 2.1 of the License.
10 //
11 //  This library is distributed in the hope that it will be useful,
12 //  but WITHOUT ANY WARRANTY; without even the implied warranty of
13 //  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14 //  Lesser General Public License for more details.
15 //
16 //  You should have received a copy of the GNU Lesser General Public
17 //  License along with this library; if not, write to the Free Software
18 //  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA
19 //
20 //  See http://www.salome-platform.org/ or email : webmaster.salome@opencascade.com
21 //
22 // GEOM GEOMGUI : GUI for Geometry component
23 // File   : GEOMBase_Tools.cxx
24 // Author : Damien COQUERET, Open CASCADE S.A.S.
25 //
26 #include "GEOMToolsGUI.h"
27 #include "GEOMToolsGUI_DeleteDlg.h"
28
29 #include <GeometryGUI.h>
30 #include "GeometryGUI_Operations.h"
31 #include <GEOMBase.h>
32 #include <GEOM_Operation.h>
33 #include <GEOM_Displayer.h>
34
35 #include <SUIT_Session.h>
36 #include <SUIT_OverrideCursor.h>
37 #include <SUIT_MessageBox.h>
38 #include <SUIT_Tools.h>
39 #include <SUIT_FileDlg.h>
40 #include <SUIT_Desktop.h>
41 #include <SUIT_ViewModel.h>
42 #include <SUIT_ViewManager.h>
43
44 #include <SalomeApp_Application.h>
45 #include <SalomeApp_Study.h>
46 #include <LightApp_SelectionMgr.h>
47 #include <GEOMImpl_Types.hxx>
48
49 #include <SALOME_ListIO.hxx>
50 #include <SALOME_ListIteratorOfListIO.hxx>
51 #include <SALOME_Prs.h>
52
53 // QT Includes
54 #include <QApplication>
55 #include <QMap>
56 #include <QRegExp>
57
58 // OCCT Includes
59 #include <TCollection_AsciiString.hxx>
60
61 typedef QMap<QString, QString> FilterMap;
62 static QString lastUsedFilter;
63
64 //=======================================================================
65 // function : getFileName
66 // purpose  : Selection of a file name for Import/Export. Returns also
67 //            the selected file type code through <filter> argument.
68 //=======================================================================
69 static QString getFileName( QWidget*           parent,
70                             const QString&     initial,
71                             const FilterMap&   filterMap,
72                             const QStringList& filters,
73                             const QString&     caption,
74                             bool               open,
75                             QString&           format,
76                             bool               showCurrentDirInitially = false )
77 {
78   //QStringList filters;
79   QString aBrepFilter;
80   for ( FilterMap::const_iterator it = filterMap.begin(); it != filterMap.end(); ++it ) {
81     //filters.push_back( it.key() );
82     if ( it.key().contains( "BREP", Qt::CaseInsensitive ) )
83       aBrepFilter = it.key();
84   }
85
86   SUIT_FileDlg* fd = new SUIT_FileDlg( parent, open, true, true );
87   if ( !caption.isEmpty() )
88     fd->setWindowTitle( caption );
89
90   if ( !initial.isEmpty() )
91     fd->selectFile( initial );
92   
93   if ( showCurrentDirInitially && SUIT_FileDlg::getLastVisitedPath().isEmpty() )
94     fd->setDirectory( QDir::currentPath() );
95
96   fd->setFilters( filters );
97   
98   if ( !lastUsedFilter.isEmpty() && filterMap.contains( lastUsedFilter ) ) {
99     fd->selectFilter( lastUsedFilter );
100   }
101   else if ( !aBrepFilter.isEmpty() ) {
102     fd->selectFilter( aBrepFilter );
103   }
104
105   QString filename;
106   if ( fd->exec() == QDialog::Accepted ) {
107     filename = fd->selectedFile();
108     format = filterMap[fd->selectedFilter()];
109     lastUsedFilter = fd->selectedFilter();
110   }
111
112   delete fd;
113   qApp->processEvents();
114   return filename;
115 }
116
117 //=======================================================================
118 // function : getFileNames
119 // purpose  : Select list of files for Import operation. Returns also
120 //            the selected file type code through <format> argument.
121 //=======================================================================
122 static QStringList getFileNames( QWidget*           parent,
123                                  const QString&     initial,
124                                  const FilterMap&   filterMap,
125                                  const QString&     caption,
126                                  QString&           format,
127                                  bool               showCurrentDirInitially = false)
128 {
129   QString aBrepFilter;
130   QStringList allFilters;
131   QStringList filters;
132   QRegExp re( "\\((.*)\\)" );
133   re.setMinimal( true );
134   for ( FilterMap::const_iterator it = filterMap.begin(); it != filterMap.end(); ++it ) {
135     if ( it.value().contains( "BREP", Qt::CaseInsensitive ) && aBrepFilter.isEmpty() )
136       aBrepFilter = it.key();
137     filters.append( it.key() );
138     int pos = 0;
139     while ( re.indexIn( it.key(), pos ) >= 0 ) {
140       QString f = re.cap(1);
141       pos = re.pos() + f.length() + 2;
142       allFilters.append( f.simplified() );
143     }
144   }
145   filters.append( QObject::tr( "GEOM_ALL_IMPORT_FILES" ).arg( allFilters.join( " " ) ) );
146   
147   SUIT_FileDlg fd( parent, true, true, true );
148   fd.setFileMode( SUIT_FileDlg::ExistingFiles );     
149   if ( !caption.isEmpty() )
150     fd.setWindowTitle( caption );
151   if ( !initial.isEmpty() )
152     fd.selectFile( initial );
153   
154   if ( showCurrentDirInitially && SUIT_FileDlg::getLastVisitedPath().isEmpty() )
155     fd.setDirectory( QDir::currentPath() );
156   
157   fd.setFilters( filters );
158   
159   if ( !lastUsedFilter.isEmpty() && filterMap.contains( lastUsedFilter ) )
160     fd.selectFilter( lastUsedFilter );
161   else if ( !aBrepFilter.isEmpty() )
162     fd.selectFilter( aBrepFilter );
163
164   QStringList filenames;
165   if ( fd.exec() ) {
166     filenames = fd.selectedFiles();
167     format = filterMap.contains( fd.selectedFilter() ) ? filterMap[ fd.selectedFilter() ] : QString();
168     lastUsedFilter = fd.selectedFilter();
169   }
170   qApp->processEvents();
171   return filenames;
172 }
173
174 //=======================================================================
175 // function : getParentComponent
176 // purpose  : Get object's parent component entry
177 //=======================================================================
178 static QString getParentComponent( _PTR( SObject ) obj )
179 {
180   if ( obj ) {
181     _PTR(SComponent) comp = obj->GetFatherComponent();
182     if ( comp )
183       return QString( comp->GetID().c_str() );
184   }
185   return QString();
186 }
187
188 //=====================================================================================
189 // function : inUse
190 // purpose  : check if the object(s) passed as the the second arguments are used
191 //            by the other objects in the study
192 //=====================================================================================
193 static bool inUse( _PTR(Study) study, const QString& component, const QMap<QString,QString>& objects )
194 {
195   _PTR(SObject) comp = study->FindObjectID( component.toLatin1().data() );
196   if ( !comp )
197     return false;
198
199   // collect all GEOM objects being deleted
200   QMap<QString, GEOM::GEOM_Object_var> gobjects;
201   QMap<QString, QString>::ConstIterator oit;
202   std::list<_PTR(SObject)> aSelectedSO;
203   for ( oit = objects.begin(); oit != objects.end(); ++oit ) {
204     _PTR(SObject) so = study->FindObjectID( oit.key().toLatin1().data() );
205     if ( !so )
206       continue;
207     aSelectedSO.push_back(so);
208     CORBA::Object_var corbaObj_rem = GeometryGUI::ClientSObjectToObject( so );
209     GEOM::GEOM_Object_var geomObj_rem = GEOM::GEOM_Object::_narrow( corbaObj_rem );
210     if( CORBA::is_nil( geomObj_rem ) ) 
211       continue;
212     gobjects.insert( oit.key(), geomObj_rem );
213   }
214
215   // Search References with other Modules
216   std::list< _PTR(SObject) >::iterator itSO = aSelectedSO.begin();
217   for ( ; itSO != aSelectedSO.end(); ++itSO ) {
218     std::vector<_PTR(SObject)> aReferences = study->FindDependances( *itSO  );    
219     int aRefLength = aReferences.size();
220     if (aRefLength) {
221       for (int i = 0; i < aRefLength; i++) {
222         _PTR(SObject) firstSO( aReferences[i] );
223         _PTR(SComponent) aComponent = firstSO->GetFatherComponent();
224         QString type = aComponent->ComponentDataType().c_str();
225         if ( type == "SMESH" )
226           return true;
227       }
228     }
229   }
230
231   // browse through all GEOM data tree
232   _PTR(ChildIterator) it ( study->NewChildIterator( comp ) );
233   for ( it->InitEx( true ); it->More(); it->Next() ) {
234     _PTR(SObject) child( it->Value() );
235     CORBA::Object_var corbaObj = GeometryGUI::ClientSObjectToObject( child );
236     GEOM::GEOM_Object_var geomObj = GEOM::GEOM_Object::_narrow( corbaObj );
237     if( CORBA::is_nil( geomObj ) ) 
238       continue;
239
240     GEOM::ListOfGO_var list = geomObj->GetDependency();
241     if( list->length() == 0 ) 
242       continue;
243
244     for( int i = 0; i < list->length(); i++ ) {
245       bool depends = false;
246       bool deleted = false;
247       QMap<QString, GEOM::GEOM_Object_var>::Iterator git;
248       for ( git = gobjects.begin(); git != gobjects.end() && ( !depends || !deleted ); ++git ) {
249         depends = depends || list[i]->_is_equivalent( *git );
250         deleted = deleted || git.key() == child->GetID().c_str() ;//geomObj->_is_equivalent( *git );
251       }
252       if ( depends && !deleted )
253         return true;
254     }
255   }
256   return false;
257 }
258
259
260 //=======================================================================
261 // function : GEOMToolsGUI()
262 // purpose  : Constructor
263 //=======================================================================
264 GEOMToolsGUI::GEOMToolsGUI( GeometryGUI* parent )
265 : GEOMGUI( parent )
266 {
267 }
268
269
270 //=======================================================================
271 // function : ~GEOMToolsGUI()
272 // purpose  : Destructor
273 //=======================================================================
274 GEOMToolsGUI::~GEOMToolsGUI()
275 {
276 }
277
278
279 //=======================================================================
280 // function : OnGUIEvent()
281 // purpose  :
282 //=======================================================================
283 bool GEOMToolsGUI::OnGUIEvent(int theCommandID, SUIT_Desktop* parent)
284 {
285   getGeometryGUI()->EmitSignalDeactivateDialog();
286
287   switch ( theCommandID ) {
288   case GEOMOp::OpDelete:         // EDIT - DELETE
289     OnEditDelete();
290     break;
291   case GEOMOp::OpImport:         // FILE - IMPORT
292     Import();
293     break;
294   case GEOMOp::OpExport:         // FILE - EXPORT
295     Export();
296     break;
297   case GEOMOp::OpCheckGeom:      // TOOLS - CHECK GEOMETRY
298     OnCheckGeometry();
299     break;
300   case GEOMOp::OpSelectVertex:   // POPUP - SELECT ONLY - VERTEX
301     OnSelectOnly( GEOM_POINT );
302     break;
303   case GEOMOp::OpSelectEdge:     // POPUP - SELECT ONLY - EDGE
304     OnSelectOnly( GEOM_EDGE );
305     break;
306   case GEOMOp::OpSelectWire:     // POPUP - SELECT ONLY - WIRE
307     OnSelectOnly( GEOM_WIRE );
308     break;
309   case GEOMOp::OpSelectFace:     // POPUP - SELECT ONLY - FACE
310     OnSelectOnly( GEOM_FACE );
311     break;
312   case GEOMOp::OpSelectShell:    // POPUP - SELECT ONLY - SHELL
313     OnSelectOnly( GEOM_SHELL );
314     break;
315   case GEOMOp::OpSelectSolid:    // POPUP - SELECT ONLY - SOLID
316     OnSelectOnly( GEOM_SOLID );
317     break;
318   case GEOMOp::OpSelectCompound: // POPUP - SELECT ONLY - COMPOUND
319     OnSelectOnly( GEOM_COMPOUND );
320     break;
321   case GEOMOp::OpSelectAll:      // POPUP - SELECT ONLY - SELECT ALL
322     OnSelectOnly( GEOM_ALLOBJECTS );
323     break;
324   case GEOMOp::OpRename:         // POPUP - RENAME
325     OnRename();
326     break;
327   case GEOMOp::OpDeflection:     // POPUP - DEFLECTION ANGLE
328     OnDeflection();
329     break;
330   case GEOMOp::OpColor:          // POPUP - COLOR
331     OnColor();
332     break;
333   case GEOMOp::OpTransparency:   // POPUP - TRANSPARENCY
334     OnTransparency();
335     break;
336   case GEOMOp::OpIsos:           // POPUP - ISOS
337     OnNbIsos();
338     break;
339   case GEOMOp::OpAutoColor:      // POPUP - AUTO COLOR
340     OnAutoColor();
341     break;
342   case GEOMOp::OpNoAutoColor:    // POPUP - DISABLE AUTO COLOR
343     OnDisableAutoColor();
344     break;
345   case GEOMOp::OpShowChildren:   // POPUP - SHOW CHILDREN
346   case GEOMOp::OpHideChildren:   // POPUP - HIDE CHILDREN
347     OnShowHideChildren( theCommandID == GEOMOp::OpShowChildren );
348     break;
349   case GEOMOp::OpPointMarker:    // POPUP - POINT MARKER
350     OnPointMarker();
351     break;
352   default:
353     SUIT_Session::session()->activeApplication()->putInfo(tr("GEOM_PRP_COMMAND").arg(theCommandID));
354     break;
355   }
356   return true;
357 }
358
359
360 //===============================================================================
361 // function : OnEditDelete()
362 // purpose  :
363 //===============================================================================
364 void GEOMToolsGUI::OnEditDelete()
365 {
366   SALOME_ListIO selected;
367   SalomeApp_Application* app =
368     dynamic_cast< SalomeApp_Application* >( SUIT_Session::session()->activeApplication() );
369   if ( !app )
370     return;
371
372   LightApp_SelectionMgr* aSelMgr = app->selectionMgr();
373   SalomeApp_Study* appStudy = dynamic_cast<SalomeApp_Study*>( app->activeStudy() );
374   if ( !aSelMgr || !appStudy )
375     return;
376
377   // get selection
378   aSelMgr->selectedObjects( selected, "ObjectBrowser", false );
379   if ( selected.IsEmpty() )
380     return;
381
382   _PTR(Study) aStudy = appStudy->studyDS();
383   
384   // check if study is locked
385   if ( _PTR(AttributeStudyProperties)( aStudy->GetProperties() )->IsLocked() ) {
386     SUIT_MessageBox::warning( app->desktop(),
387                               tr("WRN_WARNING"),
388                               tr("WRN_STUDY_LOCKED") );
389     return; // study is locked
390   }
391   
392   // get GEOM component
393   CORBA::String_var geomIOR = app->orb()->object_to_string( GeometryGUI::GetGeomGen() );
394   QString geomComp = getParentComponent( aStudy->FindObjectIOR( geomIOR.in() ) );
395
396   // check each selected object: if belongs to GEOM, if not reference...
397   QMap<QString,QString> toBeDeleted;
398   QMap<QString,QString> allDeleted;
399   bool isComponentSelected = false;
400   for ( SALOME_ListIteratorOfListIO It( selected ); It.More(); It.Next() ) {
401     Handle(SALOME_InteractiveObject) anIObject = It.Value();
402     if ( !anIObject->hasEntry() )
403       continue; // invalid object
404     // ...
405     QString entry = anIObject->getEntry();
406     _PTR(SObject) obj = aStudy->FindObjectID( entry.toLatin1().data() );
407     // check parent component
408     QString parentComp = getParentComponent( obj );
409     if ( parentComp != geomComp )  {
410       SUIT_MessageBox::warning( app->desktop(),
411                                 QObject::tr("ERR_ERROR"),
412                                 QObject::tr("NON_GEOM_OBJECTS_SELECTED").arg( getGeometryGUI()->moduleName() ) );
413       return; // not GEOM object selected
414     }
415
416     ///////////////////////////////////////////////////////
417     // if GEOM component is selected, so skip other checks
418     if ( isComponentSelected ) continue; 
419     ///////////////////////////////////////////////////////
420         
421     // check if object is reference
422     _PTR(SObject) refobj;
423     if ( obj && obj->ReferencedObject( refobj ) ) {
424       // get the main object by reference IPAL 21354
425       obj = refobj;
426       entry = obj->GetID().c_str();
427     }
428     // ...
429     QString aName = obj->GetName().c_str();
430     if ( entry == geomComp ) {
431       // GEOM component is selected, skip other checks
432       isComponentSelected = true;
433       continue;
434     }
435     toBeDeleted.insert( entry, aName );
436     allDeleted.insert( entry, aName ); // skip GEOM component
437     // browse through all children recursively
438     _PTR(ChildIterator) it ( aStudy->NewChildIterator( obj ) );
439     for ( it->InitEx( true ); it->More(); it->Next() ) {
440       _PTR(SObject) child( it->Value() );
441       if ( child && child->ReferencedObject( refobj ) )
442         continue; // skip references
443       aName = child->GetName().c_str();
444       if ( !aName.isEmpty() )
445         allDeleted.insert( child->GetID().c_str(), aName );
446     }
447   }
448   
449   // is there is anything to delete?
450   if ( !isComponentSelected && allDeleted.count() <= 0 )
451     return; // nothing to delete
452
453   // show confirmation dialog box
454   GEOMToolsGUI_DeleteDlg dlg( app->desktop(), allDeleted, isComponentSelected );
455   if ( !dlg.exec() )
456     return; // operation is cancelled by user
457   
458   // get currently opened views
459   QList<SALOME_View*> views;
460   SALOME_View* view;
461   ViewManagerList vmans = app->viewManagers();
462   SUIT_ViewManager* vman;
463   foreach ( vman, vmans ) {
464     SUIT_ViewModel* vmod = vman->getViewModel();
465     view = dynamic_cast<SALOME_View*> ( vmod ); // must work for OCC and VTK views
466     if ( view )
467       views.append( view );
468   }
469   
470   _PTR(StudyBuilder) aStudyBuilder (aStudy->NewBuilder());
471   GEOM_Displayer* disp = new GEOM_Displayer( appStudy );
472   
473   if ( isComponentSelected ) {
474     // GEOM component is selected: delete all objects recursively
475     _PTR(SObject) comp = aStudy->FindObjectID( geomComp.toLatin1().data() );
476     if ( !comp )
477       return;
478     _PTR(ChildIterator) it ( aStudy->NewChildIterator( comp ) );
479     // remove top-level objects only
480     for ( it->InitEx( false ); it->More(); it->Next() ) {
481       _PTR(SObject) child( it->Value() );
482       // remove object from GEOM engine
483       removeObjectWithChildren( child, aStudy, views, disp );
484       // remove object from study
485       aStudyBuilder->RemoveObjectWithChildren( child );
486     }
487   }
488   else {
489     // GEOM component is not selected: check if selected objects are in use
490     if ( inUse( aStudy, geomComp, allDeleted ) ) {
491       SUIT_MessageBox::warning( app->desktop(),
492                                 QObject::tr("WRN_WARNING"),
493                                 QObject::tr("DEP_OBJECT") );
494       return; // object(s) in use
495     }
496     // ... and then delete all objects
497     QMap<QString, QString>::Iterator it;
498     for ( it = toBeDeleted.begin(); it != toBeDeleted.end(); ++it ) {
499       _PTR(SObject) obj ( aStudy->FindObjectID( it.key().toLatin1().data() ) );
500       // remove object from GEOM engine
501       removeObjectWithChildren( obj, aStudy, views, disp );
502       // remove objects from study
503       aStudyBuilder->RemoveObjectWithChildren( obj );
504     }
505   }
506   
507   selected.Clear();
508   aSelMgr->setSelectedObjects( selected );
509   getGeometryGUI()->updateObjBrowser();
510   app->updateActions(); //SRN: To update a Save button in the toolbar
511 }
512
513 //=====================================================================================
514 // function : Import
515 // purpose  : BRep, Iges, Step, ...
516 //=====================================================================================
517 bool GEOMToolsGUI::Import()
518 {
519   SalomeApp_Application* app = dynamic_cast< SalomeApp_Application* >( getGeometryGUI()->getApp() );
520   if ( !app ) return false;
521
522   SalomeApp_Study* stud = dynamic_cast<SalomeApp_Study*> ( app->activeStudy() );
523   if ( !stud ) {
524     MESSAGE ( "FAILED to cast active study to SalomeApp_Study" );
525     return false;
526   }
527   _PTR(Study) aStudy = stud->studyDS();
528
529   // check if study is locked
530   bool aLocked = (_PTR(AttributeStudyProperties)(aStudy->GetProperties()))->IsLocked();
531   if ( aLocked ) {
532     SUIT_MessageBox::warning( app->desktop(),
533                               QObject::tr("WRN_WARNING"),
534                               QObject::tr("WRN_STUDY_LOCKED") );
535     return false;
536   }
537
538   // check if GEOM engine is available
539   GEOM::GEOM_Gen_var eng = GeometryGUI::GetGeomGen();
540   if ( CORBA::is_nil( eng ) ) {
541     SUIT_MessageBox::critical( app->desktop(),
542                                QObject::tr("WRN_WARNING"),
543                                QObject::tr( "GEOM Engine is not started" ) );
544     return false;
545   }
546
547   GEOM::GEOM_IInsertOperations_var aInsOp = eng->GetIInsertOperations( aStudy->StudyId() );
548   if ( aInsOp->_is_nil() )
549     return false;
550
551   // obtain a list of available import formats
552   FilterMap aMap;
553   GEOM::string_array_var aFormats, aPatterns;
554   aInsOp->ImportTranslators( aFormats, aPatterns );
555
556   for ( int i = 0, n = aFormats->length(); i < n; i++ )
557     aMap.insert( (char*)aPatterns[i], (char*)aFormats[i] );
558
559   // select files to be imported
560   QString fileType;
561   QStringList fileNames = getFileNames( app->desktop(), "", aMap,
562                                         tr( "GEOM_MEN_IMPORT" ), fileType, true );
563
564   // set Wait cursor
565   SUIT_OverrideCursor wc;
566
567   if ( fileNames.count() == 0 )
568     return false; // nothing selected, return
569
570   QStringList errors;
571
572   QList< GEOM::GEOM_Object_var > objsForDisplay;
573
574   
575   // iterate through all selected files
576
577   SUIT_MessageBox::StandardButton igesAnswer = SUIT_MessageBox::NoButton;
578   SUIT_MessageBox::StandardButton acisAnswer = SUIT_MessageBox::NoButton;
579
580   for ( int i = 0; i < fileNames.count(); i++ ) {
581     QString fileName = fileNames[i];
582
583     if ( fileName.isEmpty() )
584       continue;
585
586     QString aCurrentType;
587     if ( fileType.isEmpty() ) {
588       // file type is not defined, try to detect
589       QString ext = QFileInfo( fileName ).suffix().toUpper();
590       QRegExp re( "\\*\\.(\\w+)" );
591       for ( FilterMap::const_iterator it = aMap.begin(); 
592             it != aMap.end() && aCurrentType.isEmpty(); ++it ) {
593         int pos = 0;
594         while ( re.indexIn( it.key(), pos ) >= 0 ) {
595           QString f = re.cap(1).trimmed().toUpper();
596           if ( ext == f ) { aCurrentType = it.value(); break; }
597           pos = re.pos() + re.cap(1).length() + 2;
598         }
599       }
600     }
601     else {
602       aCurrentType = fileType;
603     }
604
605     if ( aCurrentType.isEmpty() ) {
606       errors.append( QString( "%1 : %2" ).arg( fileName ).arg( tr( "GEOM_UNSUPPORTED_TYPE" ) ) );
607       continue;
608     }
609
610     GEOM_Operation* anOp = new GEOM_Operation( app, aInsOp.in() );
611     try {
612       app->putInfo( tr( "GEOM_PRP_LOADING" ).arg( SUIT_Tools::file( fileName, /*withExten=*/true ) ) );
613       anOp->start();
614
615       CORBA::String_var fileN = fileName.toLatin1().constData();
616       CORBA::String_var fileT = aCurrentType.toLatin1().constData();
617
618       // skl 29.05.2009
619       if ( aCurrentType == "IGES" ) {
620         GEOM::GEOM_Object_var anObj = aInsOp->Import( fileN, "IGES_UNIT" );
621         bool needConvert = false;
622         TCollection_AsciiString aUnitName = aInsOp->GetErrorCode();
623         if ( aUnitName.SubString( 1, 4 ) == "UNIT" )
624           needConvert = aUnitName.SubString( 6, aUnitName.Length() ) != "M";
625
626         if ( needConvert ) {
627           if ( igesAnswer == SUIT_MessageBox::NoToAll ) {
628             // converting for all files is already approved
629             fileT = "IGES_SCALE";
630           }
631           else if ( igesAnswer != SUIT_MessageBox::YesToAll ) {
632             SUIT_MessageBox::StandardButtons btns = SUIT_MessageBox::Yes | SUIT_MessageBox::No;
633             if ( i < fileNames.count()-1 ) btns = btns | SUIT_MessageBox::YesToAll | SUIT_MessageBox::NoToAll;
634             igesAnswer = SUIT_MessageBox::question( app->desktop(),
635                                                     "Question",//tr("WRN_WARNING"),
636                                                     tr("GEOM_SCALE_DIMENSIONS"),
637                                                     btns | SUIT_MessageBox::Cancel,
638                                                     SUIT_MessageBox::No );
639             switch ( igesAnswer ) {
640             case SUIT_MessageBox::Cancel:
641               return false;                // cancel (break) import operation
642             case SUIT_MessageBox::Yes:
643             case SUIT_MessageBox::YesToAll:
644               break;                       // scaling is confirmed
645             case SUIT_MessageBox::No:
646             case SUIT_MessageBox::NoAll:
647               fileT = "IGES_SCALE";
648             default:
649               break;                       // scaling is rejected
650             } // switch ( igesAnswer )
651           } // if ( igeAnswer != NoToAll )
652         } // if ( needConvert )
653       } // if ( aCurrentType == "IGES" )
654       else if ( aCurrentType == "ACIS" ) {
655         if ( acisAnswer != SUIT_MessageBox::YesToAll && acisAnswer != SUIT_MessageBox::NoToAll ) {
656           SUIT_MessageBox::StandardButtons btns = SUIT_MessageBox::Yes | SUIT_MessageBox::No;
657           if ( i < fileNames.count()-1 ) btns = btns | SUIT_MessageBox::YesToAll | SUIT_MessageBox::NoToAll;
658           acisAnswer = SUIT_MessageBox::question( app->desktop(),
659                                                   "Question",//tr("WRN_WARNING"),
660                                                   tr("GEOM_PUBLISH_NAMED_SHAPES"),
661                                                   btns | SUIT_MessageBox::Cancel,
662                                                   SUIT_MessageBox::No );
663           if ( acisAnswer == SUIT_MessageBox::Cancel )
664             return false; // cancel (break) import operation
665         } // if ( acisAnswer != YesToAll && acisAnswer != NoToAll )
666       } // else if ( aCurrentType == "ACIS" )
667       
668       GEOM::GEOM_Object_var anObj = aInsOp->Import( fileN, fileT );
669
670       if ( !anObj->_is_nil() && aInsOp->IsDone() ) {
671         QString aPublishObjName = 
672           GEOMBase::GetDefaultName( SUIT_Tools::file( fileName, /*withExten=*/true ) );
673         
674         SALOMEDS::Study_var aDSStudy = GeometryGUI::ClientStudyToStudy( aStudy );
675         GeometryGUI::GetGeomGen()->PublishInStudy( aDSStudy,
676                                                    SALOMEDS::SObject::_nil(),
677                                                    anObj,
678                                                    aPublishObjName.toLatin1().constData() );
679         
680         objsForDisplay.append( anObj );
681         
682         if ( aCurrentType == "ACIS" ) {
683           if ( acisAnswer == SUIT_MessageBox::Yes || acisAnswer == SUIT_MessageBox::YesToAll )
684             GeometryGUI::GetGeomGen()->PublishNamedShapesInStudy( aDSStudy, anObj );
685         }
686
687         anOp->commit();
688       }
689       else {
690         anOp->abort();
691         errors.append( QString( "%1 : %2" ).arg( fileName ).arg( aInsOp->GetErrorCode() ) );
692       }
693     }
694     catch( const SALOME::SALOME_Exception& S_ex ) {
695       anOp->abort();
696       errors.append( QString( "%1 : %2" ).arg( fileName ).arg( tr( "GEOM_UNKNOWN_IMPORT_ERROR" ) ) );
697     }
698   }
699
700   // update object browser
701   getGeometryGUI()->updateObjBrowser( true );
702
703   // display imported model (if only one file is selected)
704   if ( objsForDisplay.count() == 1 )
705     GEOM_Displayer( stud ).Display( objsForDisplay[0].in() );
706
707   if ( errors.count() > 0 ) {
708     SUIT_MessageBox::critical( app->desktop(),
709                                QObject::tr( "GEOM_ERROR" ),
710                                QObject::tr( "GEOM_IMPORT_ERRORS" ) + "\n" + errors.join( "\n" ) );
711   }
712
713   app->updateActions(); //SRN: To update a Save button in the toolbar
714
715   return objsForDisplay.count() > 0;
716 }
717
718
719 //=====================================================================================
720 // function : Export
721 // purpose  : BRep, Iges, Step
722 //=====================================================================================
723 bool GEOMToolsGUI::Export()
724 {
725   SalomeApp_Application* app = getGeometryGUI()->getApp();
726   if (!app) return false;
727
728   SalomeApp_Study* stud = dynamic_cast<SalomeApp_Study*> ( app->activeStudy() );
729   if ( !stud ) {
730     MESSAGE ( "FAILED to cast active study to SalomeApp_Study" );
731     return false;
732   }
733   _PTR(Study) aStudy = stud->studyDS();
734
735   GEOM::GEOM_Gen_var eng = GeometryGUI::GetGeomGen();
736   if ( CORBA::is_nil( eng ) ) {
737     SUIT_MessageBox::critical( app->desktop(),
738                                QObject::tr("WRN_WARNING"),
739                                QObject::tr( "GEOM Engine is not started" ) );
740     return false;
741   }
742
743   GEOM::GEOM_IInsertOperations_var aInsOp = eng->GetIInsertOperations( aStudy->StudyId() );
744   if ( aInsOp->_is_nil() )
745     return false;
746
747   // Obtain a list of available export formats
748   FilterMap aMap;
749   QStringList filters;
750   GEOM::string_array_var aFormats, aPatterns;
751   aInsOp->ExportTranslators( aFormats, aPatterns );
752   for ( int i = 0, n = aFormats->length(); i < n; i++ ) {
753     aMap.insert( (char*)aPatterns[i], (char*)aFormats[i] );
754     filters.push_back( (char*)aPatterns[i] );
755   }
756
757   // Get selected objects
758   LightApp_SelectionMgr* sm = app->selectionMgr();
759   if ( !sm )
760     return false;
761
762   SALOME_ListIO selectedObjects;
763   sm->selectedObjects( selectedObjects );
764   bool appropriateObj = false;
765
766   SALOME_ListIteratorOfListIO It( selectedObjects );
767   for(;It.More();It.Next()) {
768     Handle(SALOME_InteractiveObject) IObject = It.Value();
769     Standard_Boolean found;
770     GEOM::GEOM_Object_var anObj = GEOMBase::ConvertIOinGEOMObject(IObject, found);
771
772     if ( !found || anObj->_is_nil() )
773       continue;
774
775     QString fileType;
776     QString file = getFileName(app->desktop(), QString( IObject->getName() ), aMap, filters,
777                                tr("GEOM_MEN_EXPORT"), false, fileType, true);
778
779     // User has pressed "Cancel" --> stop the operation
780     if ( file.isEmpty() || fileType.isEmpty() )
781       return false;
782
783     GEOM_Operation* anOp = new GEOM_Operation( app, aInsOp.in() );
784     try {
785       SUIT_OverrideCursor wc;
786
787       app->putInfo( tr("GEOM_PRP_EXPORT").arg(SUIT_Tools::file( file, /*withExten=*/true )) );
788
789       anOp->start();
790
791
792       aInsOp->Export( anObj, file.toStdString().c_str(), fileType.toLatin1().constData() );
793
794       if ( aInsOp->IsDone() )
795         anOp->commit();
796       else
797         {
798           anOp->abort();
799           wc.suspend();
800           SUIT_MessageBox::critical( app->desktop(),
801                                      QObject::tr( "GEOM_ERROR" ),
802                                      QObject::tr("GEOM_PRP_ABORT") + "\n" + QString( aInsOp->GetErrorCode() ) );
803           return false;
804         }
805     }
806     catch (const SALOME::SALOME_Exception& S_ex) {
807       //QtCatchCorbaException(S_ex);
808       anOp->abort();
809       return false;
810     }
811     appropriateObj = true;
812   }
813
814   if ( !appropriateObj )
815     SUIT_MessageBox::warning( app->desktop(),
816                               QObject::tr("WRN_WARNING"),
817                               QObject::tr("GEOM_WRN_NO_APPROPRIATE_SELECTION") );
818   return true;
819 }
820
821 //=====================================================================================
822 // function : RemoveObjectWithChildren
823 // purpose  : used by OnEditDelete() method
824 //=====================================================================================
825 void GEOMToolsGUI::removeObjectWithChildren(_PTR(SObject) obj,
826                                             _PTR(Study) aStudy,
827                                             QList<SALOME_View*> views,
828                                             GEOM_Displayer* disp)
829 {
830   // iterate through all children of obj
831   for (_PTR(ChildIterator) it (aStudy->NewChildIterator(obj)); it->More(); it->Next()) {
832     _PTR(SObject) child (it->Value());
833     removeObjectWithChildren(child, aStudy, views, disp);
834   }
835
836   // erase object and remove it from engine
837   _PTR(GenericAttribute) anAttr;
838   if (obj->FindAttribute(anAttr, "AttributeIOR")) {
839     _PTR(AttributeIOR) anIOR (anAttr);
840
841     // Delete shape in Client
842     const TCollection_AsciiString ASCIor ((char*)anIOR->Value().c_str());
843     getGeometryGUI()->GetShapeReader().RemoveShapeFromBuffer(ASCIor);
844
845     CORBA::Object_var corbaObj = GeometryGUI::ClientSObjectToObject(obj);
846     GEOM::GEOM_Object_var geomObj = GEOM::GEOM_Object::_narrow( corbaObj );
847     if (!CORBA::is_nil(geomObj)) {
848       // Erase graphical object
849       QListIterator<SALOME_View*> it( views );
850       while ( it.hasNext() )
851         if ( SALOME_View* view = it.next() )
852           disp->Erase(geomObj, true, view);
853       
854       // Remove object from Engine
855       // We can't directly remove object from engine. All we can do is to unpublish the object
856       // from the study. Another client could be using the object.
857       // Unpublishing is done just after in aStudyBuilder->RemoveObjectWithChildren( child );
858       //GeometryGUI::GetGeomGen()->RemoveObject( geomObj );
859     }
860   }
861 }
862
863 //=================================================================================
864 // function : deactivate()
865 // purpose  : Called when GEOM component is deactivated
866 //=================================================================================
867 void GEOMToolsGUI::deactivate()
868 {
869   SalomeApp_Application* app = dynamic_cast< SalomeApp_Application* >( SUIT_Session::session()->activeApplication() );
870   if ( app ) {
871     SalomeApp_Study* appStudy = dynamic_cast<SalomeApp_Study*>( app->activeStudy() );
872     GEOM_Displayer aDisp (appStudy);
873     aDisp.GlobalSelection();
874     getGeometryGUI()->setLocalSelectionMode(GEOM_ALLOBJECTS);
875   }
876 }
877
878 //=====================================================================================
879 // EXPORTED METHODS
880 //=====================================================================================
881 extern "C"
882 {
883 #ifdef WIN32
884   __declspec( dllexport )
885 #endif
886   GEOMGUI* GetLibGUI( GeometryGUI* parent )
887   {
888     return new GEOMToolsGUI( parent );
889   }
890 }