Salome HOME
0020577: EDF 1164 SMESH: Bad dump of concatenate with create common groups
[modules/smesh.git] / src / SMESHGUI / SMESHGUI_RemoveElementsDlg.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 //  SMESH SMESHGUI : GUI for SMESH component
23 //  File   : SMESHGUI_RemoveElementsDlg.cxx
24 //  Author : Nicolas REJNERI
25 //  Module : SMESH
26 //  $Header$
27 //
28 #include "SMESHGUI_RemoveElementsDlg.h"
29
30 #include "SMESHGUI.h"
31 #include "SMESHGUI_Utils.h"
32 #include "SMESHGUI_VTKUtils.h"
33 #include "SMESHGUI_MeshUtils.h"
34 #include "SMESHGUI_IdValidator.h"
35
36 #include "SMESH_Actor.h"
37 #include "SMDS_Mesh.hxx"
38
39 #include "SUIT_ResourceMgr.h"
40 #include "SUIT_Desktop.h"
41 #include "SUIT_Session.h"
42 #include "SUIT_MessageBox.h"
43
44 #include "SVTK_Selector.h"
45 #include "SVTK_ViewModel.h"
46 #include "SVTK_ViewWindow.h"
47 #include "SALOME_ListIO.hxx"
48
49 #include "SalomeApp_Tools.h"
50 #include "LightApp_Application.h"
51 #include "utilities.h"
52
53 // OCCT Includes
54 #include <TColStd_MapOfInteger.hxx>
55 #include <TColStd_IndexedMapOfInteger.hxx>
56
57 // QT Includes
58 #include <qbuttongroup.h>
59 #include <qgroupbox.h>
60 #include <qlabel.h>
61 #include <qlineedit.h>
62 #include <qpushbutton.h>
63 #include <qradiobutton.h>
64 #include <qlayout.h>
65 #include <qvariant.h>
66 #include <qtooltip.h>
67 #include <qwhatsthis.h>
68 #include <qimage.h>
69 #include <qpixmap.h>
70
71 using namespace std;
72
73 #include CORBA_SERVER_HEADER(SMESH_MeshEditor)
74
75 //=================================================================================
76 // class    : SMESHGUI_RemoveElementsDlg()
77 // purpose  :
78 //=================================================================================
79 SMESHGUI_RemoveElementsDlg
80 ::SMESHGUI_RemoveElementsDlg (SMESHGUI* theModule, 
81                               const char* name,
82                               bool modal, 
83                               WFlags fl)
84   : QDialog(SMESH::GetDesktop(theModule), 
85             name, 
86             modal, 
87             WStyle_Customize | WStyle_NormalBorder | WStyle_Title | WStyle_SysMenu | Qt::WDestructiveClose),
88     mySelector(SMESH::GetViewWindow(theModule)->GetSelector()),
89     mySelectionMgr(SMESH::GetSelectionMgr(theModule)),
90     mySMESHGUI(theModule),
91     myBusy(false)
92 {
93     QPixmap image0 (SMESH::GetResourceMgr( mySMESHGUI )->loadPixmap("SMESH", tr("ICON_DLG_REM_ELEMENT")));
94     QPixmap image1 (SMESH::GetResourceMgr( mySMESHGUI )->loadPixmap("SMESH", tr("ICON_SELECT")));
95
96     if (!name)
97       setName("SMESHGUI_RemoveElementsDlg");
98     resize(303, 185);
99     setCaption(tr("SMESH_REMOVE_ELEMENTS_TITLE"));
100     setSizeGripEnabled(TRUE);
101     SMESHGUI_RemoveElementsDlgLayout = new QGridLayout(this);
102     SMESHGUI_RemoveElementsDlgLayout->setSpacing(6);
103     SMESHGUI_RemoveElementsDlgLayout->setMargin(11);
104
105     /***************************************************************/
106     GroupConstructors = new QButtonGroup(this, "GroupConstructors");
107     GroupConstructors->setTitle(tr("SMESH_ELEMENTS" ));
108     GroupConstructors->setExclusive(TRUE);
109     GroupConstructors->setColumnLayout(0, Qt::Vertical);
110     GroupConstructors->layout()->setSpacing(0);
111     GroupConstructors->layout()->setMargin(0);
112     GroupConstructorsLayout = new QGridLayout(GroupConstructors->layout());
113     GroupConstructorsLayout->setAlignment(Qt::AlignTop);
114     GroupConstructorsLayout->setSpacing(6);
115     GroupConstructorsLayout->setMargin(11);
116     Constructor1 = new QRadioButton(GroupConstructors, "Constructor1");
117     Constructor1->setText(tr("" ));
118     Constructor1->setPixmap(image0);
119     Constructor1->setChecked(TRUE);
120     Constructor1->setSizePolicy(QSizePolicy((QSizePolicy::SizeType)1, (QSizePolicy::SizeType)0, Constructor1->sizePolicy().hasHeightForWidth()));
121     Constructor1->setMinimumSize(QSize(50, 0));
122     GroupConstructorsLayout->addWidget(Constructor1, 0, 0);
123     QSpacerItem* spacer = new QSpacerItem(20, 20, QSizePolicy::Expanding, QSizePolicy::Minimum);
124     GroupConstructorsLayout->addItem(spacer, 0, 1);
125     SMESHGUI_RemoveElementsDlgLayout->addWidget(GroupConstructors, 0, 0);
126
127     /***************************************************************/
128     GroupButtons = new QGroupBox(this, "GroupButtons");
129     GroupButtons->setGeometry(QRect(10, 10, 281, 48));
130     GroupButtons->setTitle(tr("" ));
131     GroupButtons->setColumnLayout(0, Qt::Vertical);
132     GroupButtons->layout()->setSpacing(0);
133     GroupButtons->layout()->setMargin(0);
134     GroupButtonsLayout = new QGridLayout(GroupButtons->layout());
135     GroupButtonsLayout->setAlignment(Qt::AlignTop);
136     GroupButtonsLayout->setSpacing(6);
137     GroupButtonsLayout->setMargin(11);
138     buttonHelp = new QPushButton(GroupButtons, "buttonHelp");
139     buttonHelp->setText(tr("SMESH_BUT_HELP" ));
140     buttonHelp->setAutoDefault(TRUE);
141     GroupButtonsLayout->addWidget(buttonHelp, 0, 4);
142     buttonCancel = new QPushButton(GroupButtons, "buttonCancel");
143     buttonCancel->setText(tr("SMESH_BUT_CLOSE" ));
144     buttonCancel->setAutoDefault(TRUE);
145     GroupButtonsLayout->addWidget(buttonCancel, 0, 3);
146     buttonApply = new QPushButton(GroupButtons, "buttonApply");
147     buttonApply->setText(tr("SMESH_BUT_APPLY" ));
148     buttonApply->setAutoDefault(TRUE);
149     GroupButtonsLayout->addWidget(buttonApply, 0, 1);
150     QSpacerItem* spacer_9 = new QSpacerItem(20, 20, QSizePolicy::Expanding, QSizePolicy::Minimum);
151     GroupButtonsLayout->addItem(spacer_9, 0, 2);
152     buttonOk = new QPushButton(GroupButtons, "buttonOk");
153     buttonOk->setText(tr("SMESH_BUT_APPLY_AND_CLOSE" ));
154     buttonOk->setAutoDefault(TRUE);
155     buttonOk->setDefault(TRUE);
156     GroupButtonsLayout->addWidget(buttonOk, 0, 0);
157     SMESHGUI_RemoveElementsDlgLayout->addWidget(GroupButtons, 2, 0);
158
159     /***************************************************************/
160     GroupC1 = new QGroupBox(this, "GroupC1");
161     GroupC1->setTitle(tr("SMESH_REMOVE" ));
162     GroupC1->setMinimumSize(QSize(0, 0));
163     GroupC1->setFrameShape(QGroupBox::Box);
164     GroupC1->setFrameShadow(QGroupBox::Sunken);
165     GroupC1->setColumnLayout(0, Qt::Vertical);
166     GroupC1->layout()->setSpacing(0);
167     GroupC1->layout()->setMargin(0);
168     GroupC1Layout = new QGridLayout(GroupC1->layout());
169     GroupC1Layout->setAlignment(Qt::AlignTop);
170     GroupC1Layout->setSpacing(6);
171     GroupC1Layout->setMargin(11);
172     TextLabelC1A1 = new QLabel(GroupC1, "TextLabelC1A1");
173     TextLabelC1A1->setText(tr("SMESH_ID_ELEMENTS" ));
174     TextLabelC1A1->setMinimumSize(QSize(50, 0));
175     TextLabelC1A1->setFrameShape(QLabel::NoFrame);
176     TextLabelC1A1->setFrameShadow(QLabel::Plain);
177     GroupC1Layout->addWidget(TextLabelC1A1, 0, 0);
178     SelectButtonC1A1 = new QPushButton(GroupC1, "SelectButtonC1A1");
179     SelectButtonC1A1->setText(tr("" ));
180     SelectButtonC1A1->setPixmap(image1);
181     SelectButtonC1A1->setToggleButton(FALSE);
182     GroupC1Layout->addWidget(SelectButtonC1A1, 0, 1);
183     LineEditC1A1 = new QLineEdit(GroupC1, "LineEditC1A1");
184     LineEditC1A1->setValidator(new SMESHGUI_IdValidator(this, "validator"));
185     GroupC1Layout->addWidget(LineEditC1A1, 0, 2);
186     SMESHGUI_RemoveElementsDlgLayout->addWidget(GroupC1, 1, 0);
187
188     myHelpFileName = "removing_nodes_and_elements_page.html#removing_elements_anchor";
189
190     Init(); /* Initialisations */
191 }
192
193 //=================================================================================
194 // function : ~SMESHGUI_RemoveElementsDlg()
195 // purpose  : Destroys the object and frees any allocated resources
196 //=================================================================================
197 SMESHGUI_RemoveElementsDlg::~SMESHGUI_RemoveElementsDlg()
198 {
199   // no need to delete child widgets, Qt does it all for us
200 }
201
202 //=================================================================================
203 // function : Init()
204 // purpose  :
205 //=================================================================================
206 void SMESHGUI_RemoveElementsDlg::Init()
207 {
208   GroupC1->show();
209   myConstructorId = 0;
210   Constructor1->setChecked(TRUE);
211   myEditCurrentArgument = LineEditC1A1;
212
213   myNbOkElements = false;
214   mySMESHGUI->SetActiveDialogBox((QDialog*)this);
215   myActor = 0;
216   myBusy = false;
217
218   /* signals and slots connections */
219   connect(buttonOk, SIGNAL(clicked()),     this, SLOT(ClickOnOk()));
220   connect(buttonCancel, SIGNAL(clicked()), this, SLOT(ClickOnCancel()));
221   connect(buttonApply, SIGNAL(clicked()), this, SLOT(ClickOnApply()));
222   connect(buttonHelp, SIGNAL(clicked()), this, SLOT(ClickOnHelp()));
223   connect(GroupConstructors, SIGNAL(clicked(int)), SLOT(ConstructorsClicked(int)));
224
225   connect(SelectButtonC1A1, SIGNAL (clicked()),   this, SLOT(SetEditCurrentArgument()));
226   connect(mySMESHGUI, SIGNAL (SignalDeactivateActiveDialog()), this, SLOT(DeactivateActiveDialog()));
227   connect(mySelectionMgr, SIGNAL(currentSelectionChanged()), this, SLOT(SelectionIntoArgument()));
228   /* to close dialog if study change */
229   connect(mySMESHGUI, SIGNAL (SignalCloseAllDialogs()), this, SLOT(ClickOnCancel()));
230   connect(myEditCurrentArgument, SIGNAL(textChanged(const QString&)),
231            SLOT(onTextChange(const QString&)));
232
233   this->show(); /* displays Dialog */
234
235   if ( SVTK_ViewWindow* aViewWindow = SMESH::GetViewWindow( mySMESHGUI ))
236     aViewWindow->SetSelectionMode(CellSelection);
237
238   SelectionIntoArgument();
239 }
240
241 //=================================================================================
242 // function : ConstructorsClicked()
243 // purpose  : Radio button management
244 //=================================================================================
245 void SMESHGUI_RemoveElementsDlg::ConstructorsClicked (int)
246 {
247 }
248
249 //=================================================================================
250 // function : ClickOnApply()
251 // purpose  :
252 //=================================================================================
253 void SMESHGUI_RemoveElementsDlg::ClickOnApply()
254 {
255   if (mySMESHGUI->isActiveStudyLocked())
256     return;
257   if (myNbOkElements) {
258     QStringList aListId = QStringList::split(" ", myEditCurrentArgument->text(), false);
259     SMESH::long_array_var anArrayOfIdeces = new SMESH::long_array;
260     anArrayOfIdeces->length(aListId.count());
261     for (int i = 0; i < aListId.count(); i++)
262       anArrayOfIdeces[i] = aListId[ i ].toInt();
263
264     bool aResult = false;
265     try {
266       SMESH::SMESH_MeshEditor_var aMeshEditor = myMesh->GetMeshEditor();
267       aResult = aMeshEditor->RemoveElements(anArrayOfIdeces.inout());
268     } catch (const SALOME::SALOME_Exception& S_ex) {
269       SalomeApp_Tools::QtCatchCorbaException(S_ex);
270       myEditCurrentArgument->clear();
271     } catch (...){
272       myEditCurrentArgument->clear();
273     }
274
275     if (aResult) {
276       myEditCurrentArgument->clear();
277       mySelector->ClearIndex();
278       SMESH::UpdateView();
279     }
280   }
281 }
282
283 //=================================================================================
284 // function : ClickOnOk()
285 // purpose  :
286 //=================================================================================
287 void SMESHGUI_RemoveElementsDlg::ClickOnOk()
288 {
289   this->ClickOnApply();
290   this->ClickOnCancel();
291
292   return;
293 }
294
295 //=================================================================================
296 // function : ClickOnCancel()
297 // purpose  :
298 //=================================================================================
299 void SMESHGUI_RemoveElementsDlg::ClickOnCancel()
300 {
301   //mySelectionMgr->clearSelected();
302   if ( SVTK_ViewWindow* aViewWindow = SMESH::GetViewWindow( mySMESHGUI ))
303     aViewWindow->SetSelectionMode(ActorSelection);
304   disconnect(mySelectionMgr, 0, this, 0);
305   mySMESHGUI->ResetState();
306   reject();
307   return;
308 }
309
310 //=================================================================================
311 // function : ClickOnHelp()
312 // purpose  :
313 //=================================================================================
314 void SMESHGUI_RemoveElementsDlg::ClickOnHelp()
315 {
316   LightApp_Application* app = (LightApp_Application*)(SUIT_Session::session()->activeApplication());
317   if (app) 
318     app->onHelpContextModule(mySMESHGUI ? app->moduleName(mySMESHGUI->moduleName()) : QString(""), myHelpFileName);
319   else {
320                 QString platform;
321 #ifdef WIN32
322                 platform = "winapplication";
323 #else
324                 platform = "application";
325 #endif
326     SUIT_MessageBox::warn1(0, QObject::tr("WRN_WARNING"),
327                            QObject::tr("EXTERNAL_BROWSER_CANNOT_SHOW_PAGE").
328                            arg(app->resourceMgr()->stringValue("ExternalBrowser", platform)).arg(myHelpFileName),
329                            QObject::tr("BUT_OK"));
330   }
331 }
332
333 //=======================================================================
334 //function : onTextChange
335 //purpose  :
336 //=======================================================================
337 void SMESHGUI_RemoveElementsDlg::onTextChange (const QString& theNewText)
338 {
339   if (myBusy) 
340     return;
341   myBusy = true;
342
343   myNbOkElements = 0;
344
345   buttonOk->setEnabled(false);
346   buttonApply->setEnabled(false);
347
348   // hilight entered elements
349   if(myActor){
350     if(SMDS_Mesh* aMesh = myActor->GetObject()->GetMesh()){
351       Handle(SALOME_InteractiveObject) anIO = myActor->getIO();
352       
353       TColStd_MapOfInteger newIndices;
354       
355       QStringList aListId = QStringList::split(" ", theNewText, false);
356       for (int i = 0; i < aListId.count(); i++) {
357         if(const SMDS_MeshElement *anElem = aMesh->FindElement(aListId[i].toInt())) {
358           newIndices.Add(anElem->GetID());
359           myNbOkElements++;
360         }
361       }
362       
363       mySelector->AddOrRemoveIndex(anIO,newIndices,false);
364       if ( SVTK_ViewWindow* aViewWindow = SMESH::GetViewWindow( mySMESHGUI ))
365         aViewWindow->highlight(anIO,true,true);
366     }
367   }
368   
369   if (myNbOkElements) {
370     buttonOk->setEnabled(true);
371     buttonApply->setEnabled(true);
372   }
373   
374   myBusy = false;
375 }
376
377 //=================================================================================
378 // function : SelectionIntoArgument()
379 // purpose  : Called when selection as changed or other case
380 //=================================================================================
381 void SMESHGUI_RemoveElementsDlg::SelectionIntoArgument()
382 {
383   if (myBusy) return;
384
385   // clear
386
387   myNbOkElements = false;
388   myActor = 0;
389
390   myBusy = true;
391   myEditCurrentArgument->setText("");
392   myBusy = false;
393
394   if (!GroupButtons->isEnabled()) // inactive
395     return;
396
397   buttonOk->setEnabled(false);
398   buttonApply->setEnabled(false);
399
400   // get selected mesh
401
402   SALOME_ListIO aList;
403   mySelectionMgr->selectedObjects(aList,SVTK_Viewer::Type());
404
405   int nbSel = aList.Extent();
406   if (nbSel != 1)
407     return;
408
409   Handle(SALOME_InteractiveObject) anIO = aList.First();
410   myMesh = SMESH::GetMeshByIO(anIO);
411   if (myMesh->_is_nil())
412     return;
413
414   myActor = SMESH::FindActorByEntry(anIO->getEntry());
415   if (!myActor)
416     return;
417
418   // get selected nodes
419   QString aString = "";
420   int nbElems = SMESH::GetNameOfSelectedElements(mySelector,anIO,aString);
421   if(nbElems < 1)
422     return;
423   myBusy = true;
424   myEditCurrentArgument->setText(aString);
425   myBusy = false;
426
427   // OK
428
429   myNbOkElements = nbElems;
430
431   buttonOk->setEnabled(true);
432   buttonApply->setEnabled(true);
433 }
434
435 //=================================================================================
436 // function : SetEditCurrentArgument()
437 // purpose  :
438 //=================================================================================
439 void SMESHGUI_RemoveElementsDlg::SetEditCurrentArgument()
440 {
441   QPushButton* send = (QPushButton*)sender();
442   switch (myConstructorId)
443     {
444     case 0: /* default constructor */
445       {
446         if(send == SelectButtonC1A1) {
447           LineEditC1A1->setFocus();
448           myEditCurrentArgument = LineEditC1A1;
449         }
450         SelectionIntoArgument();
451         break;
452       }
453     }
454   return;
455 }
456
457 //=================================================================================
458 // function : DeactivateActiveDialog()
459 // purpose  :
460 //=================================================================================
461 void SMESHGUI_RemoveElementsDlg::DeactivateActiveDialog()
462 {
463   if (GroupConstructors->isEnabled()) {
464     GroupConstructors->setEnabled(false);
465     GroupC1->setEnabled(false);
466     GroupButtons->setEnabled(false);
467     mySMESHGUI->ResetState(); // ??
468     mySMESHGUI->SetActiveDialogBox(0); // ??
469   }
470 }
471
472 //=================================================================================
473 // function : ActivateThisDialog()
474 // purpose  :
475 //=================================================================================
476 void SMESHGUI_RemoveElementsDlg::ActivateThisDialog()
477 {
478   /* Emit a signal to deactivate the active dialog */
479   mySMESHGUI->EmitSignalDeactivateDialog();
480
481   GroupConstructors->setEnabled(true);
482   GroupC1->setEnabled(true);
483   GroupButtons->setEnabled(true);
484
485   mySMESHGUI->SetActiveDialogBox((QDialog*)this); // ??
486
487   if ( SVTK_ViewWindow* aViewWindow = SMESH::GetViewWindow( mySMESHGUI ))
488     aViewWindow->SetSelectionMode(NodeSelection);
489
490   SelectionIntoArgument(); // ??
491 }
492
493 //=================================================================================
494 // function : enterEvent()
495 // purpose  :
496 //=================================================================================
497 void SMESHGUI_RemoveElementsDlg::enterEvent (QEvent*)
498 {
499   if (!GroupConstructors->isEnabled())
500     ActivateThisDialog();
501 }
502
503 //=================================================================================
504 // function : closeEvent()
505 // purpose  :
506 //=================================================================================
507 void SMESHGUI_RemoveElementsDlg::closeEvent (QCloseEvent*)
508 {
509   /* same than click on cancel button */
510   this->ClickOnCancel();
511   return;
512 }
513
514 //=======================================================================
515 //function : hideEvent
516 //purpose  : caused by ESC key
517 //=======================================================================
518 void SMESHGUI_RemoveElementsDlg::hideEvent (QHideEvent * e)
519 {
520   if (!isMinimized())
521     ClickOnCancel();
522 }
523
524 //=================================================================================
525 // function : keyPressEvent()
526 // purpose  :
527 //=================================================================================
528 void SMESHGUI_RemoveElementsDlg::keyPressEvent( QKeyEvent* e )
529 {
530   QDialog::keyPressEvent( e );
531   if ( e->isAccepted() )
532     return;
533
534   if ( e->key() == Key_F1 )
535     {
536       e->accept();
537       ClickOnHelp();
538     }
539 }