Salome HOME
e9ead7a401794c4996ddec9e5843a36e77ccbfd8
[modules/geom.git] / src / GenerationGUI / GenerationGUI_PipeDlg.cxx
1 // Copyright (C) 2007-2014  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, or (at your option) any later version.
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 // GEOM GEOMGUI : GUI for Geometry component
24 // File   : GenerationGUI_PipeDlg.cxx
25 // Author : Lucien PIGNOLONI, Open CASCADE S.A.S.
26 //
27 #include "GenerationGUI_PipeDlg.h"
28
29 #include <DlgRef.h>
30 #include <GeometryGUI.h>
31 #include <GEOMBase.h>
32
33 #include <SUIT_Session.h>
34 #include <SUIT_ResourceMgr.h>
35 #include <SalomeApp_Application.h>
36 #include <LightApp_SelectionMgr.h>
37
38 #include <TopoDS_Shape.hxx>
39 #include <TopoDS.hxx>
40 #include <TopExp.hxx>
41 #include <TColStd_IndexedMapOfInteger.hxx>
42 #include <TopTools_IndexedMapOfShape.hxx>
43 #include <TColStd_IndexedMapOfInteger.hxx>
44 #include <TColStd_MapOfInteger.hxx>
45
46 #include <GEOMImpl_Types.hxx>
47
48 //=================================================================================
49 // class    : GenerationGUI_PipeDlg()
50 // purpose  : Constructs a GenerationGUI_PipeDlg which is a child of 'parent', with the
51 //            name 'name' and widget flags set to 'f'.
52 //            The dialog will by default be modeless, unless you set 'modal' to
53 //            TRUE to construct a modal dialog.
54 //=================================================================================
55 GenerationGUI_PipeDlg::GenerationGUI_PipeDlg (GeometryGUI* theGeometryGUI, QWidget* parent,
56                                               bool modal, Qt::WindowFlags fl)
57   : GEOMBase_Skeleton(theGeometryGUI, parent, modal, fl)
58 {
59   QPixmap image0 (SUIT_Session::session()->resourceMgr()->loadPixmap("GEOM", tr("ICON_DLG_PIPE")));
60   QPixmap image1 (SUIT_Session::session()->resourceMgr()->loadPixmap("GEOM", tr("ICON_SELECT")));
61   QPixmap image2 (SUIT_Session::session()->resourceMgr()->loadPixmap("GEOM", tr("ICON_DLG_PIPE_BINORMAL")));
62   QPixmap image3 (SUIT_Session::session()->resourceMgr()->loadPixmap("GEOM", tr("ICON_DLG_PIPE_SECTION")));
63
64   setWindowTitle(tr("GEOM_PIPE_TITLE"));
65
66   /***************************************************************/
67   mainFrame()->GroupConstructors->setTitle(tr("GEOM_PIPE"));
68   mainFrame()->RadioButton1->setIcon(image0);
69   mainFrame()->RadioButton2->setIcon(image2);
70   mainFrame()->RadioButton3->setIcon(image3);
71   // mainFrame()->RadioButton3->setAttribute(Qt::WA_DeleteOnClose);
72   // mainFrame()->RadioButton3->close();
73
74   GroupPoints = new DlgRef_3Sel1Check(centralWidget());
75
76   GroupPoints->GroupBox1->setTitle(tr("GEOM_ARGUMENTS"));
77   GroupPoints->TextLabel1->setText(tr("GEOM_BASE_OBJECT"));
78   GroupPoints->TextLabel2->setText(tr("GEOM_PATH_OBJECT"));
79   GroupPoints->TextLabel3->setText(tr("GEOM_BINORMAL"));
80   GroupPoints->PushButton1->setIcon(image1);
81   GroupPoints->PushButton2->setIcon(image1);
82   GroupPoints->PushButton3->setIcon(image1);
83   GroupPoints->CheckButton1->setText(tr("GEOM_SELECT_UNPUBLISHED_EDGES"));
84
85   GroupMakePoints = new DlgRef_3Sel2Check3Spin(centralWidget());
86
87   GroupMakePoints->GroupBox1->setTitle(tr("GEOM_ARGUMENTS"));
88   GroupMakePoints->TextLabel1->setText(tr("GEOM_BASE_OBJECT"));
89   GroupMakePoints->TextLabel2->setText(tr("GEOM_LOCATIONS"));
90   GroupMakePoints->TextLabel3->setText(tr("GEOM_PATH_OBJECT"));
91   GroupMakePoints->PushButton1->setIcon(image1);
92   GroupMakePoints->PushButton2->setIcon(image1);
93   GroupMakePoints->PushButton3->setIcon(image1);
94   GroupMakePoints->CheckBox1->setText(tr("GEOM_WITH_CONTACT"));
95   GroupMakePoints->CheckBox2->setText(tr("GEOM_WITH_CORRECTION"));
96   GroupMakePoints->SpinBox1->close();
97   GroupMakePoints->SpinBox2->close();
98   GroupMakePoints->SpinBox3->close();
99   GroupMakePoints->TextLabel4->close();
100   GroupMakePoints->TextLabel5->close();
101   GroupMakePoints->TextLabel6->close();
102
103   QVBoxLayout* layout = new QVBoxLayout(centralWidget());
104   layout->setMargin(0); layout->setSpacing(6);
105   layout->addWidget(GroupPoints);
106   layout->addWidget(GroupMakePoints);
107   /***************************************************************/
108
109   setHelpFileName("create_extrusion_alongpath_page.html");
110
111   // Initialisation
112   Init();
113 }
114
115 //=================================================================================
116 // function : ~GenerationGUI_PipeDlg()
117 // purpose  : Destroys the object and frees any allocated resources
118 //=================================================================================
119 GenerationGUI_PipeDlg::~GenerationGUI_PipeDlg()
120 {
121   // no need to delete child widgets, Qt does it all for us
122 }
123
124 //=================================================================================
125 // function : Init()
126 // purpose  :
127 //=================================================================================
128 void GenerationGUI_PipeDlg::Init()
129 {
130   // init variables
131   GroupPoints->LineEdit1->setReadOnly(true);
132   GroupPoints->LineEdit2->setReadOnly(true);
133   GroupPoints->LineEdit3->setReadOnly(true);
134
135   GroupPoints->LineEdit1->setText("");
136   GroupPoints->LineEdit2->setText("");
137   GroupPoints->LineEdit3->setText("");
138   myPath.nullify();
139   myVec.nullify();
140
141   GroupPoints->CheckButton1->setEnabled(false);
142
143   GroupMakePoints->LineEdit1->setReadOnly(true);
144   GroupMakePoints->LineEdit2->setReadOnly(true);
145   GroupMakePoints->LineEdit3->setReadOnly(true);
146
147   GroupMakePoints->LineEdit1->setText("");
148   GroupMakePoints->LineEdit2->setText("");
149   GroupMakePoints->LineEdit3->setText("");
150
151   showOnlyPreviewControl();
152
153   // signals and slots connections
154   connect(buttonOk(),    SIGNAL(clicked()), this, SLOT(ClickOnOk()));
155   connect(buttonApply(), SIGNAL(clicked()), this, SLOT(ClickOnApply()));
156
157   connect(this,          SIGNAL(constructorsClicked(int)), this, SLOT(ConstructorsClicked(int)));
158
159   connect(GroupPoints->PushButton1, SIGNAL(clicked()), this, SLOT(SetEditCurrentArgument()));
160   connect(GroupPoints->PushButton2, SIGNAL(clicked()), this, SLOT(SetEditCurrentArgument()));
161   connect(GroupPoints->PushButton3, SIGNAL(clicked()), this, SLOT(SetEditCurrentArgument()));
162
163   connect(GroupPoints->CheckButton1,   SIGNAL(toggled(bool)), this, SLOT(SelectionTypeButtonClicked()));
164
165   connect(GroupMakePoints->PushButton1, SIGNAL(clicked()), this, SLOT(SetEditCurrentArgument()));
166   connect(GroupMakePoints->PushButton2, SIGNAL(clicked()), this, SLOT(SetEditCurrentArgument()));
167   connect(GroupMakePoints->PushButton3, SIGNAL(clicked()), this, SLOT(SetEditCurrentArgument()));
168
169   initName(tr("GEOM_PIPE"));
170   resize(100,100);
171
172   GroupPoints->TextLabel3->hide();
173   GroupPoints->PushButton3->hide();
174   GroupPoints->LineEdit3->hide();
175   ConstructorsClicked(0);
176
177   GroupPoints->PushButton1->click();
178   SelectionIntoArgument();
179 }
180
181 //=================================================================================
182 // function : ConstructorsClicked()
183 // purpose  : Radio button management
184 //=================================================================================
185 void GenerationGUI_PipeDlg::ConstructorsClicked( int constructorId )
186 {
187   erasePreview();
188
189   switch (constructorId) {
190   case 0:
191     GroupMakePoints->hide();
192     GroupPoints->show();
193     GroupPoints->TextLabel3->hide();
194     GroupPoints->PushButton3->hide();
195     GroupPoints->LineEdit3->hide();
196     GroupPoints->PushButton1->click();
197     break;
198   case 1:
199     GroupMakePoints->hide();
200     GroupPoints->show();
201     GroupPoints->TextLabel3->show();
202     GroupPoints->PushButton3->show();
203     GroupPoints->LineEdit3->show();
204     GroupPoints->PushButton1->click();
205     break;
206   case 2:
207     GroupPoints->hide();
208     GroupMakePoints->show();
209     GroupMakePoints->PushButton1->click();
210     break;
211   default:
212     break;
213   }
214
215   qApp->processEvents();
216   updateGeometry();
217   resize(100,100);
218
219   SelectionIntoArgument();
220   processPreview();
221 }
222
223 //=================================================================================
224 // function : SelectionBittonClicked()
225 // purpose  : Selection type Radio button management
226 //=================================================================================
227 void GenerationGUI_PipeDlg::SelectionTypeButtonClicked()
228 {
229   globalSelection();
230   if ( GroupPoints->CheckButton1->isChecked() ) {
231     localSelection( GEOM::GEOM_Object::_nil(), TopAbs_EDGE );
232   } else {
233     TColStd_MapOfInteger aMap;
234     aMap.Add(GEOM_COMPOUND);
235     aMap.Add(GEOM_WIRE);
236     aMap.Add(GEOM_LINE);
237     globalSelection(aMap);
238   }
239   if ( myEditCurrentArgument == GroupPoints->LineEdit2 ) {
240     myEditCurrentArgument->setText("");
241     myPath.nullify();
242   }
243   processPreview();
244 }
245
246 //=================================================================================
247 // function : ClickOnOk()
248 // purpose  :
249 //=================================================================================
250 void GenerationGUI_PipeDlg::ClickOnOk()
251 {
252   setIsApplyAndClose( true );
253   if (ClickOnApply())
254     ClickOnCancel();
255 }
256
257 //=================================================================================
258 // function : ClickOnApply()
259 // purpose  :
260 //=================================================================================
261 bool GenerationGUI_PipeDlg::ClickOnApply()
262 {
263   if (!onAccept())
264     return false;
265
266   initName();
267   if ( getConstructorId() != 1 )
268     ConstructorsClicked( getConstructorId() );
269   // activate selection and connect selection manager
270   //   GroupPoints->PushButton1->click();
271   return true;
272 }
273
274 //=================================================================================
275 // function : SelectionIntoArgument()
276 // purpose  : Called when selection is changed or on dialog initialization or activation
277 //=================================================================================
278 void GenerationGUI_PipeDlg::SelectionIntoArgument()
279 {
280   erasePreview();
281   myEditCurrentArgument->setText("");
282
283   if ( myEditCurrentArgument == GroupPoints->LineEdit1 ) {
284     myBaseObjects.clear();
285     QList<GEOM::GeomObjPtr> objects = getSelected( TopAbs_SHAPE, -1 );
286     for ( int i = 0; i < objects.count(); i++ ) {
287       GEOM::shape_type stype = objects[i]->GetMaxShapeType();
288       GEOM::shape_type maxallowed = getConstructorId() == 1 ? GEOM::FACE : GEOM::SHELL;
289       GEOM::shape_type minallowed = getConstructorId() == 1 ? GEOM::EDGE : GEOM::VERTEX;
290       if ( stype < maxallowed || stype > minallowed )
291         continue;
292       myBaseObjects << objects[i];
293     }
294     if ( !myBaseObjects.isEmpty() ) {
295       QString aName = myBaseObjects.count() > 1 ? QString( "%1_objects").arg( myBaseObjects.count() ) : GEOMBase::GetName( myBaseObjects[0].get() );
296       myEditCurrentArgument->setText( aName );
297     }
298   }
299   else if (myEditCurrentArgument == GroupPoints->LineEdit2) {
300     QList<TopAbs_ShapeEnum> types;
301     types << TopAbs_EDGE << TopAbs_WIRE;
302     myPath = getSelected( types );
303     if ( myPath ) {
304       QString aName = GEOMBase::GetName( myPath.get() );
305       myEditCurrentArgument->setText( aName );
306       if ( getConstructorId() == 1 && !myVec )
307         GroupPoints->PushButton3->click();
308       else if ( myBaseObjects.isEmpty() )
309         GroupPoints->PushButton1->click();
310     }
311   }
312   else if (myEditCurrentArgument == GroupPoints->LineEdit3) {
313     myVec = getSelected( TopAbs_EDGE );
314     if ( myVec ) {
315       QString aName = GEOMBase::GetName( myVec.get() );
316       myEditCurrentArgument->setText( aName );
317       if ( myBaseObjects.isEmpty() )
318         GroupPoints->PushButton1->click();
319       else if ( !myPath )
320         GroupPoints->PushButton2->click();
321     }
322   }
323   else if ( myEditCurrentArgument == GroupMakePoints->LineEdit1 ) {
324     myBaseObjects.clear();
325     QList<TopAbs_ShapeEnum> types;
326     types << TopAbs_EDGE << TopAbs_WIRE << TopAbs_FACE << TopAbs_SHELL;
327     QList<GEOM::GeomObjPtr> objects = getSelected( types, -1 );
328     GEOMBase::Synchronize( myBaseObjects, objects );
329     if ( !myBaseObjects.isEmpty() ) {
330       QString aName = myBaseObjects.count() > 1 ? QString( "%1_objects").arg( myBaseObjects.count() ) : GEOMBase::GetName( myBaseObjects[0].get() );
331       myEditCurrentArgument->setText( aName );
332     }
333   }
334   else if ( myEditCurrentArgument == GroupMakePoints->LineEdit2 ) {
335     myLocations.clear();
336     localSelection( GEOM::GEOM_Object::_nil(), TopAbs_VERTEX );
337     QList<GEOM::GeomObjPtr> objects = getSelected( TopAbs_VERTEX, -1 );
338     GEOMBase::Synchronize( myLocations, objects );
339     if ( !myLocations.isEmpty() ) {
340       QString aName = myLocations.count() > 1 ? QString( "%1_objects").arg( myLocations.count() ) : GEOMBase::GetName( myLocations[0].get() );
341       myEditCurrentArgument->setText( aName );
342     }
343   }
344   else if ( myEditCurrentArgument == GroupMakePoints->LineEdit3 ) {
345     QList<TopAbs_ShapeEnum> types;
346     types << TopAbs_EDGE << TopAbs_WIRE;
347     myPath = getSelected( types );
348     if ( myPath ) {
349       QString aName = GEOMBase::GetName( myPath.get() );
350       myEditCurrentArgument->setText( aName );
351     }
352   }
353
354   processPreview();
355 }
356
357 //=================================================================================
358 // function : SetEditCurrentArgument()
359 // purpose  :
360 //=================================================================================
361 void GenerationGUI_PipeDlg::SetEditCurrentArgument()
362 {
363   QPushButton* send = (QPushButton*)sender();
364
365   disconnect(myGeomGUI->getApp()->selectionMgr(), 0, this, 0);
366   globalSelection(GEOM_ALLSHAPES);
367   GroupPoints->PushButton1->setDown(false);
368   GroupPoints->PushButton2->setDown(false);
369   GroupPoints->PushButton3->setDown(false);
370   GroupPoints->LineEdit1->setEnabled(false);
371   GroupPoints->LineEdit2->setEnabled(false);
372   GroupPoints->LineEdit3->setEnabled(false);
373   if (send == GroupPoints->PushButton1) {
374     myEditCurrentArgument = GroupPoints->LineEdit1;
375     GroupPoints->CheckButton1->setEnabled(false);
376   }
377   else if (send == GroupPoints->PushButton2) {
378     myEditCurrentArgument = GroupPoints->LineEdit2;
379
380     if ( GroupPoints->CheckButton1->isChecked() ) {
381       localSelection( GEOM::GEOM_Object::_nil(), TopAbs_EDGE );
382     } else {
383       TColStd_MapOfInteger aMap;
384       aMap.Add(GEOM_COMPOUND);
385       aMap.Add(GEOM_WIRE);
386       aMap.Add(GEOM_LINE);
387       globalSelection(aMap);
388     }
389     GroupPoints->CheckButton1->setEnabled(true);
390   }
391   else if(send == GroupPoints->PushButton3) {
392     myEditCurrentArgument = GroupPoints->LineEdit3;
393     GroupPoints->CheckButton1->setEnabled(false);
394     localSelection(GEOM::GEOM_Object::_nil(), TopAbs_EDGE);
395   }
396
397   GroupMakePoints->PushButton1->setDown(false);
398   GroupMakePoints->PushButton2->setDown(false);
399   GroupMakePoints->PushButton3->setDown(false);
400   GroupMakePoints->LineEdit1->setEnabled(false);
401   GroupMakePoints->LineEdit2->setEnabled(false);
402   GroupMakePoints->LineEdit3->setEnabled(false);
403   if (send == GroupMakePoints->PushButton1) {
404     myEditCurrentArgument = GroupMakePoints->LineEdit1;
405   }
406   else if(send == GroupMakePoints->PushButton2) {
407     myEditCurrentArgument = GroupMakePoints->LineEdit2;
408   }
409   else if (send == GroupMakePoints->PushButton3) {
410     myEditCurrentArgument = GroupMakePoints->LineEdit3;
411   }
412
413   connect(myGeomGUI->getApp()->selectionMgr(), SIGNAL(currentSelectionChanged()),
414           this, SLOT(SelectionIntoArgument()));
415
416   // enable line edit
417   myEditCurrentArgument->setEnabled(true);
418   myEditCurrentArgument->setFocus();
419   // after setFocus(), because it will be setDown(false) when loses focus
420   send->setDown(true);
421
422   // seems we need it only to avoid preview disappearing, caused by selection mode change
423   processPreview();
424 }
425
426 //=================================================================================
427 // function : ActivateThisDialog()
428 // purpose  :
429 //=================================================================================
430 void GenerationGUI_PipeDlg::ActivateThisDialog()
431 {
432   GEOMBase_Skeleton::ActivateThisDialog();
433   connect( myGeomGUI->getApp()->selectionMgr(), SIGNAL( currentSelectionChanged() ),
434            this, SLOT( SelectionIntoArgument() ) );
435
436   ConstructorsClicked(getConstructorId());
437 }
438
439 //=================================================================================
440 // function : enterEvent()
441 // purpose  : when mouse enter onto the QWidget
442 //=================================================================================
443 void GenerationGUI_PipeDlg::enterEvent (QEvent*)
444 {
445   if (!mainFrame()->GroupConstructors->isEnabled())
446     ActivateThisDialog();
447 }
448
449 //=================================================================================
450 // function : createOperation
451 // purpose  :
452 //=================================================================================
453 GEOM::GEOM_IOperations_ptr GenerationGUI_PipeDlg::createOperation()
454 {
455   return getGeomEngine()->GetI3DPrimOperations(getStudyId());
456 }
457
458 //=================================================================================
459 // function : isValid
460 // purpose  :
461 //=================================================================================
462 bool GenerationGUI_PipeDlg::isValid (QString&)
463 {
464   bool ok = false;
465   switch ( getConstructorId() ) {
466   case 0 :
467     ok = !myBaseObjects.isEmpty() && myPath;
468     break;
469   case 1 :
470     ok = !myBaseObjects.isEmpty() && myPath && myVec;
471     break;
472   case 2 :
473     ok = !myBaseObjects.isEmpty() && ( myLocations.isEmpty() || myBaseObjects.count() == myLocations.count() ) && myPath;
474     break;
475   default:
476     break;
477   }
478   return ok;
479 }
480
481 //=================================================================================
482 // function : execute
483 // purpose  :
484 //=================================================================================
485 bool GenerationGUI_PipeDlg::execute (ObjectList& objects)
486 {
487   GEOM::GEOM_Object_var anObj;
488
489   GEOM::GEOM_I3DPrimOperations_var anOper = GEOM::GEOM_I3DPrimOperations::_narrow(getOperation());
490   switch( getConstructorId() ) {
491   case 0:
492   case 1:
493     for (int i = 0; i < myBaseObjects.count(); i++) {
494       switch ( getConstructorId() ) {
495       case 0 :
496         anObj = anOper->MakePipe(myBaseObjects[i].get(), myPath.get());
497         break;
498       case 1 :
499         anObj = anOper->MakePipeBiNormalAlongVector(myBaseObjects[i].get(), myPath.get(), myVec.get());
500         break;
501       }
502     
503       if (!anObj->_is_nil())
504         objects.push_back(anObj._retn());
505     }
506     break;
507   case 2:
508     {
509       GEOM::ListOfGO_var myBaseGO = new GEOM::ListOfGO();
510       GEOM::ListOfGO_var myLocationsGO = new GEOM::ListOfGO();
511       myBaseGO->length( myBaseObjects.count() );
512       myLocationsGO->length( myLocations.count() );
513       for (int i = 0; i < myBaseObjects.count(); i++) {
514         myBaseGO[i] = myBaseObjects[i].copy();
515       }
516       for (int i = 0; i < myLocations.count(); i++) {
517         myLocationsGO[i] = myLocations[i].copy();
518       }
519       
520       anObj = anOper->MakePipeWithDifferentSections(myBaseGO.in(), myLocationsGO.in(), myPath.get(), 
521                                                     GroupMakePoints->CheckBox1->isChecked(), 
522                                                     GroupMakePoints->CheckBox2->isChecked());
523       if (!anObj->_is_nil())
524         objects.push_back(anObj._retn());
525     }
526     break;
527   default:
528     break;
529   }
530   return true;
531 }
532
533 //=================================================================================
534 // function : addSubshapeToStudy
535 // purpose  : virtual method to add new SubObjects if local selection
536 //=================================================================================
537 void GenerationGUI_PipeDlg::addSubshapesToStudy()
538 {
539   GEOMBase::PublishSubObject( myPath.get() );
540   GEOMBase::PublishSubObject( myVec.get() );
541 }
542
543 //=================================================================================
544 // function : extractPrefix
545 // purpose  : redefined from GEOMBase_Helper class
546 //=================================================================================
547 bool GenerationGUI_PipeDlg::extractPrefix() const
548 {
549   return myBaseObjects.count() > 1;
550 }