Salome HOME
Fix for IPAL10428.
[modules/geom.git] / src / GroupGUI / GroupGUI_GroupDlg.cxx
1 //  GEOM GEOMGUI : GUI for Geometry component
2 //
3 //  Copyright (C) 2004  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 //
23 //
24 //  File   : GroupGUI_GroupDlg.cxx
25 //  Author : Sergey ANIKIN
26 //  Module : GEOM
27 //  $Header$
28
29 #include "GroupGUI_GroupDlg.h"
30
31 #include "SUIT_Desktop.h"
32 #include "SUIT_Session.h"
33 #include "SalomeApp_Application.h"
34 #include "SalomeApp_Study.h"
35 #include "LightApp_SelectionMgr.h"
36
37 #include "GEOMBase.h"
38
39 #include "GEOMImpl_Types.hxx"
40
41 #include <qlabel.h>
42 #include <qlistbox.h>
43 #include <qlineedit.h>
44 #include <qmap.h>
45
46 #include <TColStd_IndexedMapOfInteger.hxx>
47 #include <TColStd_MapOfInteger.hxx>
48
49
50 GroupGUI_GroupDlg::GroupGUI_GroupDlg(Mode mode, GeometryGUI* theGeometryGUI, QWidget* parent)
51   :GEOMBase_Skeleton( theGeometryGUI, parent, "GroupGUI_GroupDlg", false,
52                       WStyle_Customize | WStyle_NormalBorder | WStyle_Title | WStyle_SysMenu),
53    myMode( mode ),
54    myBusy( false )
55 {
56   SUIT_ResourceMgr* resMgr = SUIT_Session::session()->resourceMgr();
57
58   QPixmap image0     (resMgr->loadPixmap("GEOM", tr("ICON_OBJBROWSER_VERTEX")));
59   QPixmap image1     (resMgr->loadPixmap("GEOM", tr("ICON_OBJBROWSER_EDGE")));
60   QPixmap image2     (resMgr->loadPixmap("GEOM", tr("ICON_OBJBROWSER_FACE")));
61   QPixmap image3     (resMgr->loadPixmap("GEOM", tr("ICON_OBJBROWSER_SOLID")));
62   QPixmap iconSelect (resMgr->loadPixmap("GEOM", tr("ICON_SELECT")));
63
64   setCaption( myMode == CreateGroup ? tr( "CREATE_GROUP_TITLE" ) : tr( "EDIT_GROUP_TITLE" ) );
65
66   // Shape type button group
67   GroupConstructors->setEnabled( myMode == CreateGroup );
68   GroupConstructors->setTitle( tr( "SHAPE_TYPE" ) );
69   RadioButton1->setPixmap( image0 );
70   RadioButton2->setPixmap( image1 );
71   RadioButton3->setPixmap( image2 );
72   RadioButton4->setPixmap( image3 );
73   RadioButton4->show();
74
75   // Group name
76   GroupBoxName->setTitle( tr( "GROUP_NAME" ) );
77
78   // Main shape and sub-shapes
79   GroupMedium = new QGroupBox( 1, Qt::Vertical, tr( "MAIN_SUB_SHAPES" ), this );
80   GroupMedium->setInsideMargin( 10 );
81   Layout1->addWidget( GroupMedium, 2, 0 );
82
83   QWidget* aFrame = new QWidget( GroupMedium );
84   QGridLayout* aMedLayout = new QGridLayout( aFrame, 4, 4, 0, 6 );
85
86   QLabel* aMainLabel = new QLabel( tr( "MAIN_SHAPE" ), aFrame );
87
88   mySelBtn = new QPushButton( aFrame );
89   mySelBtn->setPixmap( iconSelect );
90   mySelBtn->setEnabled( myMode == CreateGroup );
91
92   myMainName = new QLineEdit( aFrame );
93   myMainName->setReadOnly( true );
94   myMainName->setEnabled( myMode == CreateGroup );
95
96   mySelSubBtn = new QPushButton( tr( "SELECT_SUB_SHAPES" ), aFrame );
97   mySelAllBtn = new QPushButton( tr( "SELECT_ALL" ), aFrame );
98   myAddBtn    = new QPushButton( tr( "ADD" ), aFrame );
99   myRemBtn    = new QPushButton( tr( "REMOVE" ), aFrame );
100   myIdList    = new QListBox( aFrame );
101
102   myIdList->setSelectionMode( QListBox::Extended );
103   myIdList->setRowMode( QListBox::FitToWidth );
104
105   aMedLayout->addWidget( aMainLabel, 0, 0 );
106   aMedLayout->addWidget( mySelBtn, 0, 1 );
107   aMedLayout->addMultiCellWidget( myMainName, 0, 0, 2, 3 );
108   aMedLayout->addMultiCellWidget( mySelSubBtn, 1, 1, 0, 2 );
109   aMedLayout->addWidget( mySelAllBtn, 1, 3 );
110   aMedLayout->addMultiCellWidget( myIdList, 2, 3, 0, 2 );
111   aMedLayout->addWidget( myAddBtn, 2, 3 );
112   aMedLayout->addWidget( myRemBtn, 3, 3 );
113
114   setHelpFileName("working_with_groups.htm");
115
116   Init();
117 }
118
119 GroupGUI_GroupDlg::~GroupGUI_GroupDlg()
120 {
121 }
122
123
124 //=================================================================================
125 // function : Init()
126 // purpose  :
127 //=================================================================================
128 void GroupGUI_GroupDlg::Init()
129 {
130   // san -- TODO: clear selected sub-shapes...
131
132   if ( myMode == CreateGroup ) {
133     initName( tr( "GROUP_PREFIX" ) );
134
135     // Get ready for main shape selection
136     myEditCurrentArgument = myMainName;
137
138     connect( GroupConstructors, SIGNAL( clicked( int ) ), this, SLOT( ConstructorsClicked( int ) ) );
139     connect( mySelBtn,          SIGNAL( clicked() ),      this, SLOT( SetEditCurrentArgument() ) );
140   }
141   else if ( myMode == EditGroup && IObjectCount() ) {
142     Standard_Boolean aResult = Standard_False;
143     GEOM::GEOM_Object_var anObj =
144       GEOMBase::ConvertIOinGEOMObject( firstIObject(), aResult );
145
146     if ( aResult && !CORBA::is_nil( anObj ) && anObj->GetType() == GEOM_GROUP ) {
147       myGroup = anObj;
148
149       ResultName->setText( GEOMBase::GetName( myGroup ) );
150
151       GEOM::GEOM_IGroupOperations_var anOp = GEOM::GEOM_IGroupOperations::_narrow( getOperation() );
152       myMainObj = anOp->GetMainShape( myGroup );
153       if ( !CORBA::is_nil( myMainObj ) )
154         myMainName->setText( GEOMBase::GetName( myMainObj ) );
155
156       setShapeType( (TopAbs_ShapeEnum)anOp->GetType( myGroup ) );
157
158       GEOM::ListOfLong_var aCurrList = anOp->GetObjects( myGroup );
159       QListBoxItem* anItem;
160       for ( int i = 0, n = aCurrList->length(); i < n; i++ ) {
161         anItem = new QListBoxText( QString( "%1" ).arg(aCurrList[i] ) );
162         myIdList->insertItem( anItem );
163       }
164
165       myEditCurrentArgument = 0;
166     }
167   }
168
169   LightApp_SelectionMgr* aSelMgr =
170     ((SalomeApp_Application*)(SUIT_Session::session()->activeApplication()))->selectionMgr();
171
172   connect( aSelMgr,     SIGNAL( currentSelectionChanged() ), this, SLOT( SelectionIntoArgument() ) );
173
174   connect( buttonOk   , SIGNAL( clicked() ), this, SLOT( ClickOnOk()    ) );
175   connect( buttonApply, SIGNAL( clicked() ), this, SLOT( ClickOnApply() ) );
176
177   connect( mySelSubBtn, SIGNAL( clicked() ), this, SLOT( SetEditCurrentArgument() ) );
178   connect( mySelAllBtn, SIGNAL( clicked() ), this, SLOT( SetEditCurrentArgument() ) );
179   connect( myAddBtn,    SIGNAL( clicked() ), this, SLOT( add() ) );
180   connect( myRemBtn,    SIGNAL( clicked() ), this, SLOT( remove() ) );
181   connect( myIdList,    SIGNAL( selectionChanged() ), this, SLOT( selectionChanged() ) );
182
183   activateSelection();
184 }
185
186 //=================================================================================
187 // function : enterEvent()
188 // purpose  :
189 //=================================================================================
190 void GroupGUI_GroupDlg::enterEvent( QEvent* e )
191 {
192   if ( !buttonCancel->isEnabled() )
193     this->ActivateThisDialog();
194 }
195
196 //=================================================================================
197 // function : ClickOnOk()
198 // purpose  :
199 //=================================================================================
200 void GroupGUI_GroupDlg::ClickOnOk()
201 {
202   if ( ClickOnApply() )
203     ClickOnCancel();
204 }
205
206 //=================================================================================
207 // function : ClickOnApply()
208 // purpose  :
209 //=================================================================================
210 bool GroupGUI_GroupDlg::ClickOnApply()
211 {
212   if ( !onAccept( myMode == CreateGroup, true ) )
213     return false;
214
215   if ( myMode == CreateGroup )
216     {
217       initName();
218       myIdList->clear();
219     }
220   else
221     activateSelection();
222   return true;
223 }
224
225
226 //=================================================================================
227 // function : ActivateThisDialog()
228 // purpose  :
229 //=================================================================================
230 void GroupGUI_GroupDlg::ActivateThisDialog()
231 {
232   GEOMBase_Skeleton::ActivateThisDialog();
233
234   connect( ((SalomeApp_Application*)(SUIT_Session::session()->activeApplication()))->selectionMgr(),
235            SIGNAL( currentSelectionChanged() ), this, SLOT( SelectionIntoArgument() ) );
236
237   activateSelection();
238 }
239
240 //=================================================================================
241 // function : LineEditReturnPressed()
242 // purpose  :
243 //=================================================================================
244 void GroupGUI_GroupDlg::LineEditReturnPressed()
245 {
246   QLineEdit* send = ( QLineEdit* )sender();
247
248   if ( send == myMainName && !myEditCurrentArgument ) {
249     myEditCurrentArgument = myMainName;
250     activateSelection();
251   }
252   else
253     GEOMBase_Skeleton::LineEditReturnPressed();
254
255   updateState();
256 }
257
258
259 //=================================================================================
260 // function : SetEditCurrentArgument()
261 // purpose  :
262 //=================================================================================
263 void GroupGUI_GroupDlg::SetEditCurrentArgument()
264 {
265   QPushButton* send = (QPushButton*)sender();
266
267   if ( send == mySelBtn )
268     myEditCurrentArgument = myMainName;
269   else if ( send == mySelSubBtn || send == mySelAllBtn )
270     myEditCurrentArgument = 0;
271
272   activateSelection();
273
274   if ( send == mySelAllBtn )
275     selectAllSubShapes();
276   else
277     updateState();
278 }
279
280
281 //=================================================================================
282 // function : SelectionIntoArgument()
283 // purpose  : Called when selection has changed
284 //=================================================================================
285 void GroupGUI_GroupDlg::SelectionIntoArgument()
286 {
287   if ( myEditCurrentArgument ) {  // Selection of a main shape is active
288     myEditCurrentArgument->setText( "" );
289     myIdList->clear();
290
291     if ( IObjectCount() == 1 ) {
292       Standard_Boolean aResult = Standard_False;
293       GEOM::GEOM_Object_var anObj =
294         GEOMBase::ConvertIOinGEOMObject( firstIObject(), aResult );
295
296       if ( aResult && !anObj->_is_nil() && GEOMBase::IsShape( anObj ) ) {
297         myMainObj = anObj;
298         myEditCurrentArgument->setText( GEOMBase::GetName( anObj ) );
299         updateState();
300         return;
301       }
302     }
303
304     myMainObj = GEOM::GEOM_Object::_nil();
305   }
306   else { // an attempt to synchronize list box selection with 3d viewer
307     if ( myBusy )
308       return;
309
310     bool isBlocked = myIdList->signalsBlocked();
311     myIdList->blockSignals( true );
312     myIdList->clearSelection();
313
314     TColStd_IndexedMapOfInteger aMapIndex;
315
316     if ( IObjectCount() == 1 ) {
317       Standard_Boolean aResult = Standard_False;
318       GEOM::GEOM_Object_var anObj =
319         GEOMBase::ConvertIOinGEOMObject( firstIObject(), aResult );
320
321       if ( aResult && !anObj->_is_nil() )
322         ((SalomeApp_Application*)(SUIT_Session::session()->activeApplication()))->
323           selectionMgr()->GetIndexes( firstIObject(), aMapIndex );
324     }
325
326     // try to find out and process the object browser selection
327     if ( !aMapIndex.Extent() ) {
328       globalSelection( GEOM_ALLSHAPES );
329       
330       GEOM::ListOfGO anObjects;
331       GEOMBase::ConvertListOfIOInListOfGO(selectedIO(), anObjects);
332       GEOM::GEOM_ILocalOperations_var aLocOp = getGeomEngine()->GetILocalOperations( getStudyId() );
333       for (int i = 0; i < anObjects.length(); i++) {
334         TopoDS_Shape aShape;
335         if ( GEOMBase::GetShape(anObjects[i], aShape, getShapeType()) ) {
336           CORBA::Long anIndex = aLocOp->GetSubShapeIndex( myMainObj, anObjects[i] );
337           if ( anIndex >= 0 )
338             aMapIndex.Add( anIndex );
339         }
340       }
341       
342       if ( !myMainObj->_is_nil() )
343         localSelection( myMainObj, getShapeType() );
344     }
345
346     if (aMapIndex.Extent() >= 1) {
347       QMap<int, int> aMap;
348       for ( int i = 0, n = myIdList->count(); i < n; i++ )
349         aMap.insert( myIdList->item( i )->text().toInt(), i );
350
351       for ( int ii = 1, nn = aMapIndex.Extent(); ii <= nn; ii++ ) {
352         if ( aMap.contains( aMapIndex( ii ) ) )
353           myIdList->setSelected( aMap[aMapIndex( ii )], true );
354       }
355     }
356     myIdList->blockSignals( isBlocked );
357   }
358
359   updateState();
360 }
361
362 //=================================================================================
363 // function : ConstructorsClicked()
364 // purpose  : Radio button management
365 //=================================================================================
366 void GroupGUI_GroupDlg::ConstructorsClicked( int constructorId )
367 {
368   myIdList->clear();
369   activateSelection();
370   updateState();
371 }
372
373 //=================================================================================
374 // function : selectAllSubShapes
375 // purpose  : 
376 //=================================================================================
377 void GroupGUI_GroupDlg::selectAllSubShapes()
378 {
379   if ( CORBA::is_nil( myMainObj ) )
380     return;
381
382   GEOM::GEOM_IShapesOperations_var aShOp = getGeomEngine()->GetIShapesOperations( getStudyId() );
383   GEOM::GEOM_ILocalOperations_var aLocOp = getGeomEngine()->GetILocalOperations( getStudyId() );
384
385   GEOM::ListOfGO_var aSubShapes = aShOp->MakeExplode( myMainObj, getShapeType(), false );
386   if ( !aShOp->IsDone() )
387     return;
388
389   bool isBlocked = myIdList->signalsBlocked();
390   myIdList->blockSignals( true );
391   myIdList->clear();
392
393   QListBoxItem* anItem;
394   for ( int i = 0, n = aSubShapes->length(); i < n; i++ ) {
395     CORBA::Long anIndex = aLocOp->GetSubShapeIndex( myMainObj, aSubShapes[i] );
396     if ( anIndex < 0 )
397       continue;
398
399     anItem = new QListBoxText( QString( "%1" ).arg( anIndex ) );
400     myIdList->insertItem( anItem );
401     myIdList->setSelected( anItem, true );
402   }
403
404   myIdList->blockSignals( isBlocked );
405   highlightSubShapes();
406   updateState();
407 }
408
409 //=================================================================================
410 // function : add
411 // purpose  : 
412 //=================================================================================
413 void GroupGUI_GroupDlg::add()
414 {
415   TColStd_MapOfInteger aMap;
416   for ( int i = 0, n = myIdList->count(); i < n; i++ )
417     aMap.Add( myIdList->item( i )->text().toInt() );
418
419   TColStd_IndexedMapOfInteger aMapIndex;
420
421   if ( IObjectCount() == 1 ) {
422     Standard_Boolean aResult = Standard_False;
423     GEOM::GEOM_Object_var anObj =
424       GEOMBase::ConvertIOinGEOMObject( firstIObject(), aResult );
425
426     if ( aResult && !anObj->_is_nil() )
427       ((SalomeApp_Application*)(SUIT_Session::session()->activeApplication()))->
428         selectionMgr()->GetIndexes( firstIObject(), aMapIndex );
429   }
430
431   // try to find out and process the object browser selection
432   if ( !aMapIndex.Extent() ) {
433     GEOM::ListOfGO anObjects;
434     GEOMBase::ConvertListOfIOInListOfGO(selectedIO(), anObjects);
435     GEOM::GEOM_ILocalOperations_var aLocOp = getGeomEngine()->GetILocalOperations( getStudyId() );
436     for (int i = 0; i < anObjects.length(); i++) {
437       TopoDS_Shape aShape;
438       if ( GEOMBase::GetShape(anObjects[i], aShape, getShapeType()) ) {
439         CORBA::Long anIndex = aLocOp->GetSubShapeIndex( myMainObj, anObjects[i] );
440         if ( anIndex >= 0 )
441           aMapIndex.Add( anIndex );
442       }
443     }
444   }
445
446   if ( aMapIndex.Extent() >= 1 ) {
447     QListBoxItem* anItem;
448     bool isBlocked = myIdList->signalsBlocked();
449     myIdList->blockSignals( true );
450
451     for ( int i = 1, n = aMapIndex.Extent(); i <= n; i++ ) {
452       if ( aMap.Contains( aMapIndex( i ) ) )
453         continue;
454
455       anItem = new QListBoxText( QString( "%1" ).arg( aMapIndex( i ) ) );
456       myIdList->insertItem( anItem );
457       myIdList->setSelected( anItem, true );
458     }
459
460     myIdList->blockSignals( isBlocked );
461   }
462
463   updateState();
464 }
465
466 //=================================================================================
467 // function : remove
468 // purpose  : 
469 //=================================================================================
470 void GroupGUI_GroupDlg::remove()
471 {
472   for ( int i = myIdList->count() - 1; i >= 0; i-- ) {
473     if ( myIdList->isSelected( i ) )
474       myIdList->removeItem( i );
475   }
476   updateState();
477 }
478
479
480 //=================================================================================
481 // function : getConstructorId()
482 // purpose  :
483 //=================================================================================
484 int GroupGUI_GroupDlg::getConstructorId() const
485 {
486   return GroupConstructors->id( GroupConstructors->selected() );
487 }
488
489 //=================================================================================
490 // function : getShapeType()
491 // purpose  :
492 //=================================================================================
493 TopAbs_ShapeEnum GroupGUI_GroupDlg::getShapeType() const
494 {
495   switch ( getConstructorId() ) {
496   case 0:  return TopAbs_VERTEX;
497   case 1:  return TopAbs_EDGE;
498   case 2:  return TopAbs_FACE;
499   case 3:  return TopAbs_SOLID;
500   default: return TopAbs_SHAPE;
501   }
502 }
503
504 //=================================================================================
505 // function : setShapeType()
506 // purpose  :
507 //=================================================================================
508 void GroupGUI_GroupDlg::setShapeType( const TopAbs_ShapeEnum theType )
509 {
510   int anId = 0;
511   switch ( theType ) {
512   case TopAbs_VERTEX: anId = 0; break;
513   case TopAbs_EDGE:   anId = 1; break;
514   case TopAbs_FACE:   anId = 2; break;
515   case TopAbs_SOLID:  anId = 3; break;
516   }
517   GroupConstructors->setButton( anId );
518 }
519
520
521 //=================================================================================
522 // function : activateSelection
523 // purpose  : Activate selection in accordance with myEditCurrentArgument
524 //=================================================================================
525 void GroupGUI_GroupDlg::activateSelection()
526 {
527   globalSelection( GEOM_ALLSHAPES );
528
529   if ( !myMainObj->_is_nil() && !myEditCurrentArgument ) {
530     localSelection( myMainObj, getShapeType() );
531   }
532
533   SelectionIntoArgument();
534 }
535
536 //=================================================================================
537 // function : updateState
538 // purpose  : 
539 //=================================================================================
540 void GroupGUI_GroupDlg::updateState()
541 {
542   bool isAdd = false;
543
544   TColStd_IndexedMapOfInteger aMapIndex;
545
546   if ( IObjectCount() == 1 ) {
547     Standard_Boolean aResult = Standard_False;
548     GEOM::GEOM_Object_var anObj =
549       GEOMBase::ConvertIOinGEOMObject( firstIObject(), aResult );
550
551     if ( aResult && !anObj->_is_nil() )
552       ((SalomeApp_Application*)(SUIT_Session::session()->activeApplication()))->
553         selectionMgr()->GetIndexes( firstIObject(), aMapIndex );
554   }
555
556   // try to find out and process the object browser selection
557   if ( !aMapIndex.Extent() && !CORBA::is_nil( myMainObj ) ) {
558     isAdd = true;
559     GEOM::ListOfGO anObjects;
560     GEOMBase::ConvertListOfIOInListOfGO(selectedIO(), anObjects);
561     GEOM::GEOM_ILocalOperations_var aLocOp = getGeomEngine()->GetILocalOperations( getStudyId() );
562     for (int i = 0; i < anObjects.length(); i++) {
563       TopoDS_Shape aShape;
564       if ( GEOMBase::GetShape(anObjects[i], aShape, getShapeType()) ) {
565         CORBA::Long anIndex = aLocOp->GetSubShapeIndex( myMainObj, anObjects[i] );
566         if ( anIndex >= 0 )
567           aMapIndex.Add( anIndex );
568         else
569           isAdd = false;
570       }
571       else
572         isAdd = false;
573
574       if ( !isAdd ) {
575         aMapIndex.Clear();
576         break;
577       }
578     }
579   }
580
581   isAdd = aMapIndex.Extent() > 0;
582
583   myAddBtn->setEnabled( !myEditCurrentArgument && !CORBA::is_nil( myMainObj ) && isAdd );
584   bool hasSel = false;
585   for ( int ii = 0, nn = myIdList->count(); !hasSel && ii < nn; ii++ )
586     hasSel =  myIdList->isSelected( ii );
587   myRemBtn->setEnabled( hasSel );
588   mySelSubBtn->setEnabled( !CORBA::is_nil( myMainObj ) );
589   mySelAllBtn->setEnabled( !CORBA::is_nil( myMainObj ) );
590 }
591
592 //=================================================================================
593 // function : selectionChanged
594 // purpose  :
595 //=================================================================================
596 void GroupGUI_GroupDlg::selectionChanged()
597 {
598   highlightSubShapes();
599 }
600
601 //=================================================================================
602 // function : highlightSubShapes
603 // purpose  :
604 //=================================================================================
605 void GroupGUI_GroupDlg::highlightSubShapes()
606 {
607   if ( CORBA::is_nil( myMainObj ) )
608     return;
609
610   Standard_Boolean isOk;
611   Handle(GEOM_AISShape) aSh =
612     GEOMBase::ConvertIORinGEOMAISShape( GEOMBase::GetIORFromObject( myMainObj ), isOk, true );
613   if ( !isOk || aSh.IsNull() )
614     return;
615
616   TColStd_MapOfInteger anIds;
617
618   myBusy = true;
619
620   for ( int ii = 0, nn = myIdList->count(); ii < nn; ii++ )
621     if ( myIdList->isSelected( ii ) )
622       anIds.Add( myIdList->item( ii )->text().toInt() );
623
624   LightApp_SelectionMgr* aSelMgr =
625     ((SalomeApp_Application*)(SUIT_Session::session()->activeApplication()))->selectionMgr();
626   aSelMgr->clearSelected();
627   aSelMgr->AddOrRemoveIndex( aSh->getIO(), anIds, false );
628
629   myBusy = false;
630
631   updateState();
632 }
633
634 //=================================================================================
635 // function : createOperation
636 // purpose  :
637 //=================================================================================
638 GEOM::GEOM_IOperations_ptr GroupGUI_GroupDlg::createOperation()
639 {
640   return getGeomEngine()->GetIGroupOperations( getStudyId() );
641 }
642
643 #define RETURN_WITH_MSG(a, b) \
644   if ( !(a) ) { \
645     theMessage += (b); \
646     return false; \
647   }
648
649 //=================================================================================
650 // function : isValid()
651 // purpose  : Verify validity of input data
652 //=================================================================================
653 bool GroupGUI_GroupDlg::isValid( QString& theMessage )
654 {
655   SalomeApp_Study* study = getStudy();
656   ASSERT(study);
657   RETURN_WITH_MSG  ( !study->studyDS()->GetProperties()->IsLocked(), tr( "GEOM_STUDY_LOCKED" ) )
658
659   if ( myMode == CreateGroup ) {
660     RETURN_WITH_MSG( !CORBA::is_nil( myMainObj ), tr( "NO_MAIN_OBJ" ) )
661   }
662   else {
663     RETURN_WITH_MSG( !CORBA::is_nil( myMainObj ), tr( "NO_GROUP" ) )
664   }
665
666   QString aName (getNewObjectName());
667   RETURN_WITH_MSG  ( !aName.stripWhiteSpace().isEmpty(), tr( "EMPTY_NAME" ) )
668
669   RETURN_WITH_MSG  ( myIdList->count(), tr( "EMPTY_LIST" ) )
670   return true;
671 }
672
673 //=================================================================================
674 // function : execute
675 // purpose  :
676 //=================================================================================
677 bool GroupGUI_GroupDlg::execute( ObjectList& objects )
678 {
679   GEOM::GEOM_IGroupOperations_var anOp = GEOM::GEOM_IGroupOperations::_narrow( getOperation() );
680
681   GEOM::GEOM_Object_var aGroup;
682   if ( myMode == CreateGroup )
683     aGroup = anOp->CreateGroup( myMainObj, getShapeType() );
684   else if ( myMode == EditGroup )
685     aGroup = myGroup;
686
687   if ( CORBA::is_nil( aGroup ) || ( myMode == CreateGroup && !anOp->IsDone() ) )
688     return false;
689
690   GEOM::ListOfLong_var aCurrList = anOp->GetObjects( aGroup );
691   if ( !anOp->IsDone()  )
692     return false;
693
694   for ( int i = 0, n = aCurrList->length(); i < n; i++ ) {
695     anOp->RemoveObject( aGroup, aCurrList[i] );
696     if ( !anOp->IsDone()  )
697       return false;
698   }
699
700   for ( int ii = 0, nn = myIdList->count(); ii < nn; ii++ ) {
701     anOp->AddObject( aGroup, myIdList->item( ii )->text().toInt() );
702     if ( !anOp->IsDone()  )
703       return false;
704   }
705
706   SalomeApp_Study* study = getStudy();
707   if ( study ) {
708     string IOR = GEOMBase::GetIORFromObject( aGroup );
709     if ( IOR != "" ) {
710       _PTR(SObject) SO ( study->studyDS()->FindObjectIOR( IOR ) );
711       if ( SO ) {
712         _PTR(StudyBuilder) aBuilder (study->studyDS()->NewBuilder());
713         aBuilder->SetName( SO, getNewObjectName() );
714       }
715     }
716   }
717
718   objects.push_back( aGroup._retn() );
719
720   return true;
721 }
722
723 //================================================================
724 // Function : getFather
725 // Purpose  : Get father object for object to be added in study
726 //            ( called with addInStudy method )
727 //================================================================
728 GEOM::GEOM_Object_ptr GroupGUI_GroupDlg::getFather( GEOM::GEOM_Object_ptr theObj )
729 {
730   GEOM::GEOM_Object_var aFatherObj;
731   if ( theObj->GetType() == GEOM_GROUP ) {
732     GEOM::GEOM_IGroupOperations_var anOp = GEOM::GEOM_IGroupOperations::_narrow( getOperation() );
733     aFatherObj = anOp->GetMainShape( theObj );
734   }
735   return aFatherObj._retn();
736 }