Salome HOME
Selection problems
[modules/shaper.git] / src / XGUI / XGUI_Displayer.cpp
1 // File:        XGUI_Displayer.cpp
2 // Created:     20 Apr 2014
3 // Author:      Natalia ERMOLAEVA
4
5 #include "XGUI_Displayer.h"
6 #include "XGUI_Viewer.h"
7 #include "XGUI_Workshop.h"
8 #include "XGUI_ViewerProxy.h"
9 #include "ModuleBase_Tools.h"
10
11 #include <ModelAPI_Document.h>
12 #include <ModelAPI_Data.h>
13 #include <ModelAPI_Object.h>
14
15 #include <GeomAPI_Shape.h>
16 #include <GeomAPI_IPresentable.h>
17
18 #include <AIS_InteractiveContext.hxx>
19 #include <AIS_LocalContext.hxx>
20 #include <AIS_ListOfInteractive.hxx>
21 #include <AIS_ListIteratorOfListOfInteractive.hxx>
22 #include <AIS_DimensionSelectionMode.hxx>
23
24 #include <AIS_Shape.hxx>
25
26 #include <set>
27
28 const int MOUSE_SENSITIVITY_IN_PIXEL = 10; ///< defines the local context mouse selection sensitivity
29
30 XGUI_Displayer::XGUI_Displayer(XGUI_Workshop* theWorkshop)
31 {
32   myWorkshop = theWorkshop;
33 }
34
35 XGUI_Displayer::~XGUI_Displayer()
36 {
37 }
38
39 bool XGUI_Displayer::isVisible(ObjectPtr theObject)
40 {
41   return myResult2AISObjectMap.find(theObject) != myResult2AISObjectMap.end();
42 }
43
44 void XGUI_Displayer::display(ObjectPtr theObject, bool isUpdateViewer)
45 {
46   if (isVisible(theObject)) {
47     redisplay(theObject, isUpdateViewer);
48   } else {
49     boost::shared_ptr<GeomAPI_AISObject> anAIS;
50
51     GeomPresentablePtr aPrs = boost::dynamic_pointer_cast<GeomAPI_IPresentable>(theObject);
52     if (aPrs) {
53       anAIS = aPrs->getAISObject(boost::shared_ptr<GeomAPI_AISObject>());
54     } else {
55       ResultPtr aResult = boost::dynamic_pointer_cast<ModelAPI_Result>(theObject);
56       if (aResult) {
57         boost::shared_ptr<GeomAPI_Shape> aShapePtr = ModuleBase_Tools::shape(aResult);
58         if (aShapePtr) {
59           anAIS = boost::shared_ptr<GeomAPI_AISObject>(new GeomAPI_AISObject());
60           anAIS->createShape(aShapePtr);
61         }
62       }
63     }
64     if (anAIS)
65       display(theObject, anAIS, isUpdateViewer);
66   }
67 }
68
69 void XGUI_Displayer::display(ObjectPtr theObject,
70                              boost::shared_ptr<GeomAPI_AISObject> theAIS, bool isUpdateViewer)
71 {
72   Handle(AIS_InteractiveContext) aContext = AISContext();
73
74   Handle(AIS_InteractiveObject) anAISIO = theAIS->impl<Handle(AIS_InteractiveObject)>();
75   if (!anAISIO.IsNull()) {
76     myResult2AISObjectMap[theObject] = theAIS;
77     aContext->Display(anAISIO, isUpdateViewer);
78   }
79 }
80
81
82
83 void XGUI_Displayer::erase(ObjectPtr theObject, const bool isUpdateViewer)
84 {
85   if (!isVisible(theObject)) return;
86
87   Handle(AIS_InteractiveContext) aContext = AISContext();
88   boost::shared_ptr<GeomAPI_AISObject> anObject = myResult2AISObjectMap[theObject];
89   if (anObject)  {
90     Handle(AIS_InteractiveObject) anAIS = anObject->impl<Handle(AIS_InteractiveObject)>();
91     if (!anAIS.IsNull()) {
92       aContext->Erase(anAIS, isUpdateViewer);
93     }
94   }
95   myResult2AISObjectMap.erase(theObject);
96 }
97
98
99 /*bool XGUI_Displayer::redisplay(ObjectPtr theObject,
100                                boost::shared_ptr<GeomAPI_AISObject> theAIS,
101                                const bool isUpdateViewer)
102 {
103   bool isCreated = false;
104   Handle(AIS_InteractiveObject) anAIS = 
105     theAIS ? theAIS->impl<Handle(AIS_InteractiveObject)>() : Handle(AIS_InteractiveObject)();
106   Handle(AIS_InteractiveContext) aContext = AISContext();
107   // Open local context if there is no one
108   if (!aContext->HasOpenedContext()) {
109     aContext->ClearCurrents(false);
110     aContext->OpenLocalContext(false /use displayed objects/, true /allow shape decomposition/);
111     // set mouse sensitivity
112     //aContext->SetSensitivityMode(StdSelect_SM_WINDOW);
113     //aContext->SetPixelTolerance(MOUSE_SENSITIVITY_IN_PIXEL);
114   }
115   // display or redisplay presentation
116   boost::shared_ptr<GeomAPI_AISObject> anObj = myResult2AISObjectMap[theObject];
117   if (isVisible(theObject) && anObj && !anObj->empty()) {
118     aContext->Redisplay(anAIS, isUpdateViewer);
119     //aContext->RecomputeSelectionOnly(anAIS);
120   }
121   else {
122     myResult2AISObjectMap[theObject] = theAIS;
123     aContext->Display(anAIS, isUpdateViewer);
124     isCreated = true;
125   }
126   return isCreated;
127 }*/
128
129 void XGUI_Displayer::redisplay(ObjectPtr theObject, bool isUpdateViewer)
130 {
131   if (!isVisible(theObject))
132     return;
133
134   boost::shared_ptr<GeomAPI_AISObject> aAISObj = getAISObject(theObject);
135   GeomPresentablePtr aPrs = boost::dynamic_pointer_cast<GeomAPI_IPresentable>(theObject);
136   if (aPrs) {
137     boost::shared_ptr<GeomAPI_AISObject> aAIS_Obj = aPrs->getAISObject(aAISObj);
138     if (aAISObj && !aAIS_Obj) {
139       erase(theObject, isUpdateViewer);
140       return;
141     }
142     Handle(AIS_InteractiveObject) aAISIO = aAIS_Obj->impl<Handle(AIS_InteractiveObject)>();
143     AISContext()->Redisplay(aAISIO, isUpdateViewer);
144   } else {
145     ResultPtr aResult = boost::dynamic_pointer_cast<ModelAPI_Result>(theObject);
146     if (aResult) {
147       boost::shared_ptr<GeomAPI_Shape> aShapePtr = ModuleBase_Tools::shape(aResult);
148       if (aShapePtr) {
149         Handle(AIS_Shape) aAISShape = Handle(AIS_Shape)::DownCast(aAISObj->impl<Handle(AIS_InteractiveObject)>());
150         if (!aAISShape.IsNull()) {
151           aAISShape->Set(aShapePtr->impl<TopoDS_Shape>());
152           AISContext()->Redisplay(aAISShape, isUpdateViewer);
153         }
154       }
155     }
156   }
157 }
158
159 void XGUI_Displayer::activateInLocalContext(ObjectPtr theResult,
160                                          const std::list<int>& theModes, const bool isUpdateViewer)
161 {
162   Handle(AIS_InteractiveContext) aContext = AISContext();
163   // Open local context if there is no one
164   if (!aContext->HasOpenedContext()) {
165     aContext->ClearCurrents(false);
166     aContext->OpenLocalContext(false/*use displayed objects*/, true/*allow shape decomposition*/);
167   }
168   // display or redisplay presentation
169   Handle(AIS_InteractiveObject) anAIS;
170   if (isVisible(theResult)) {
171     boost::shared_ptr<GeomAPI_AISObject> anObj = myResult2AISObjectMap[theResult];
172     if (anObj)
173       anAIS = anObj->impl<Handle(AIS_InteractiveObject)>();
174   }
175
176   // Activate selection of objects from prs
177   if (!anAIS.IsNull()) {
178     aContext->ClearSelected(false); // ToCheck
179     aContext->Load(anAIS, -1, true/*allow decomposition*/);
180     aContext->Deactivate(anAIS);
181
182     std::list<int>::const_iterator anIt = theModes.begin(), aLast = theModes.end();
183     for (; anIt != aLast; anIt++) {
184       aContext->Activate(anAIS, (*anIt));
185     }
186   }
187
188   if (isUpdateViewer)
189     updateViewer();
190 }
191
192 void XGUI_Displayer::deactvate(ObjectPtr theObject, bool toUpdate)
193 {
194   if (isVisible(theObject)) {
195     Handle(AIS_InteractiveContext) aContext = AISContext();
196
197      boost::shared_ptr<GeomAPI_AISObject> anObj = myResult2AISObjectMap[theObject];
198      Handle(AIS_InteractiveObject) anAIS = anObj->impl<Handle(AIS_InteractiveObject)>();
199      aContext->Deactivate(anAIS);
200   }
201 }
202
203 void XGUI_Displayer::stopSelection(const QList<ObjectPtr>& theResults, const bool isStop,
204                                    const bool isUpdateViewer)
205 {
206   Handle(AIS_InteractiveContext) aContext = AISContext();
207
208   Handle(AIS_Shape) anAIS;
209   QList<ObjectPtr>::const_iterator anIt = theResults.begin(), aLast = theResults.end();
210   ObjectPtr aFeature;
211   for (; anIt != aLast; anIt++) {
212     aFeature = *anIt;
213     if (isVisible(aFeature))
214       anAIS = Handle(AIS_Shape)::DownCast(myResult2AISObjectMap[aFeature]->impl<Handle(AIS_InteractiveObject)>());
215     if (anAIS.IsNull())
216       continue;
217
218     if (isStop) {
219       QColor aColor(Qt::white);
220       anAIS->SetColor(Quantity_Color(aColor.red()/255., aColor.green()/255., aColor.blue()/255., Quantity_TOC_RGB));
221       anAIS->Redisplay();
222     }
223     else {
224       QColor aColor(Qt::red);
225       anAIS->SetColor(Quantity_Color(aColor.red()/255., aColor.green()/255., aColor.blue()/255., Quantity_TOC_RGB));
226       anAIS->Redisplay();
227     }
228   }
229   if (isUpdateViewer)
230     updateViewer();
231 }
232
233 void XGUI_Displayer::setSelected(const QList<ObjectPtr>& theResults, const bool isUpdateViewer)
234 {
235   Handle(AIS_InteractiveContext) aContext = AISContext();
236   // we need to unhighligth objects manually in the current local context
237   // in couple with the selection clear (TODO)
238   Handle(AIS_LocalContext) aLocalContext = aContext->LocalContext();
239   if (!aLocalContext.IsNull())
240     aLocalContext->UnhilightLastDetected(myWorkshop->viewer()->activeView());
241
242   aContext->ClearSelected();
243   foreach(ObjectPtr aResult, theResults) {
244     if (myResult2AISObjectMap.find(aResult) == myResult2AISObjectMap.end()) 
245       continue;
246
247     boost::shared_ptr<GeomAPI_AISObject> anObj = myResult2AISObjectMap[aResult];
248     if (anObj) {
249       Handle(AIS_InteractiveObject) anAIS = anObj->impl<Handle(AIS_InteractiveObject)>();
250       if (!anAIS.IsNull())
251         aContext->AddOrRemoveSelected(anAIS, false);
252     }
253   }
254   if (isUpdateViewer)
255     updateViewer();
256 }
257
258
259 /*void XGUI_Displayer::EraseAll(const bool isUpdateViewer)
260 {
261   Handle(AIS_InteractiveContext) ic = AISContext();
262
263   AIS_ListOfInteractive aList;
264   ic->DisplayedObjects(aList);
265   AIS_ListIteratorOfListOfInteractive anIter(aList);
266   for (; anIter.More(); anIter.Next()) {
267     if ((anIter.Value()->DynamicType() == STANDARD_TYPE(AIS_Trihedron)))
268       continue;
269
270     // erase an object
271     Handle(AIS_InteractiveObject) anIO = anIter.Value();
272     ic->Erase(anIO, false);
273   }
274   myResult2AISObjectMap.clear();
275   if (isUpdateViewer)
276     updateViewer();
277 }*/
278
279 void XGUI_Displayer::eraseDeletedResults(const bool isUpdateViewer)
280 {
281   Handle(AIS_InteractiveContext) aContext = AISContext();
282
283   ResultToAISMap::const_iterator aFIt = myResult2AISObjectMap.begin(),
284                                  aFLast = myResult2AISObjectMap.end();
285   std::list<ObjectPtr> aRemoved;
286   for (; aFIt != aFLast; aFIt++)
287   {
288     ObjectPtr aFeature = (*aFIt).first;
289     if (!aFeature || !aFeature->data() || !aFeature->data()->isValid()) {
290       boost::shared_ptr<GeomAPI_AISObject> anObj = (*aFIt).second;
291       if (!anObj) continue;
292       Handle(AIS_InteractiveObject) anAIS = anObj->impl<Handle(AIS_InteractiveObject)>();
293       if (!anAIS.IsNull()) {
294         aContext->Erase(anAIS, false);
295         aRemoved.push_back(aFeature);
296       }
297     }
298   }
299   std::list<ObjectPtr>::const_iterator anIt = aRemoved.begin(),
300                                                                  aLast = aRemoved.end();
301   for (; anIt != aLast; anIt++) {
302     myResult2AISObjectMap.erase(myResult2AISObjectMap.find(*anIt));
303   }
304
305   if (isUpdateViewer)
306     updateViewer();
307 }
308
309 void XGUI_Displayer::closeLocalContexts(const bool isUpdateViewer)
310 {
311   closeAllContexts(true);
312 }
313
314 boost::shared_ptr<GeomAPI_AISObject> XGUI_Displayer::getAISObject(ObjectPtr theObject) const
315 {
316   boost::shared_ptr<GeomAPI_AISObject> anIO;
317   if (myResult2AISObjectMap.find(theObject) != myResult2AISObjectMap.end())
318     anIO = (myResult2AISObjectMap.find(theObject))->second;
319   return anIO;
320 }
321
322 ObjectPtr XGUI_Displayer::getObject(Handle(AIS_InteractiveObject) theIO) const
323 {
324   ObjectPtr aFeature;
325   ResultToAISMap::const_iterator aFIt = myResult2AISObjectMap.begin(),
326                                  aFLast = myResult2AISObjectMap.end();
327   for (; aFIt != aFLast && !aFeature; aFIt++) {
328     boost::shared_ptr<GeomAPI_AISObject> anObj = (*aFIt).second;
329     if (!anObj) continue;
330     Handle(AIS_InteractiveObject) anAIS = anObj->impl<Handle(AIS_InteractiveObject)>();
331     if (anAIS != theIO)
332       continue;
333     aFeature = (*aFIt).first;
334   }
335   return aFeature;
336 }
337
338 void XGUI_Displayer::closeAllContexts(const bool isUpdateViewer)
339 {
340   Handle(AIS_InteractiveContext) ic = AISContext();
341   if (!ic.IsNull()) {
342     ic->CloseAllContexts(false);
343     if (isUpdateViewer)
344       updateViewer();
345   }
346 }
347
348 void XGUI_Displayer::updateViewer()
349 {
350   Handle(AIS_InteractiveContext) ic = AISContext();
351   if (!ic.IsNull())
352     ic->UpdateCurrentViewer();
353 }
354
355 Handle(AIS_InteractiveContext) XGUI_Displayer::AISContext() const 
356
357   return myWorkshop->viewer()->AISContext(); 
358 }
359
360
361 void XGUI_Displayer::display(boost::shared_ptr<GeomAPI_AISObject> theAIS, bool isUpdate)
362 {
363   Handle(AIS_InteractiveContext) aContext = AISContext();
364   Handle(AIS_InteractiveObject) anAISIO = theAIS->impl<Handle(AIS_InteractiveObject)>();
365   if (!anAISIO.IsNull())
366     aContext->Display(anAISIO, isUpdate);
367 }
368
369 void XGUI_Displayer::erase(boost::shared_ptr<GeomAPI_AISObject> theAIS, const bool isUpdate)
370 {
371   Handle(AIS_InteractiveContext) aContext = AISContext();
372   Handle(AIS_InteractiveObject) anAISIO = theAIS->impl<Handle(AIS_InteractiveObject)>();
373   if (!anAISIO.IsNull()) {
374     aContext->Remove(anAISIO, isUpdate);
375   }
376 }
377