Salome HOME
704d5b2decf2f24ca6c50e2f68c37995719ce547
[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   Handle(AIS_InteractiveObject) aAISIO;
135   boost::shared_ptr<GeomAPI_AISObject> aAISObj = getAISObject(theObject);
136   GeomPresentablePtr aPrs = boost::dynamic_pointer_cast<GeomAPI_IPresentable>(theObject);
137   if (aPrs) {
138     boost::shared_ptr<GeomAPI_AISObject> aAIS_Obj = aPrs->getAISObject(aAISObj);
139     if (aAISObj && !aAIS_Obj) {
140       erase(theObject, isUpdateViewer);
141       return;
142     }
143     aAISIO = aAIS_Obj->impl<Handle(AIS_InteractiveObject)>();
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           aAISIO = aAISShape;
153         }
154       }
155     }
156   }
157   if (!aAISIO.IsNull()) {
158     Handle(AIS_InteractiveContext) aContext = AISContext();
159     aContext->Redisplay(aAISIO, isUpdateViewer);
160     //if (aContext->HasOpenedContext()) {
161     //  aContext->Load(aAISIO, -1, true/*allow decomposition*/);
162     //}
163   }
164 }
165
166 void XGUI_Displayer::activateInLocalContext(ObjectPtr theResult,
167                                          const std::list<int>& theModes, const bool isUpdateViewer)
168 {
169   Handle(AIS_InteractiveContext) aContext = AISContext();
170   // Open local context if there is no one
171   if (!aContext->HasOpenedContext()) {
172     aContext->ClearCurrents(false);
173     aContext->OpenLocalContext(false/*use displayed objects*/, true/*allow shape decomposition*/);
174   }
175   // display or redisplay presentation
176   Handle(AIS_InteractiveObject) anAIS;
177   if (isVisible(theResult)) {
178     boost::shared_ptr<GeomAPI_AISObject> anObj = myResult2AISObjectMap[theResult];
179     if (anObj)
180       anAIS = anObj->impl<Handle(AIS_InteractiveObject)>();
181   }
182
183   // Activate selection of objects from prs
184   if (!anAIS.IsNull()) {
185     aContext->ClearSelected(false); // ToCheck
186     aContext->Load(anAIS, -1, true/*allow decomposition*/);
187     aContext->Deactivate(anAIS);
188
189     std::list<int>::const_iterator anIt = theModes.begin(), aLast = theModes.end();
190     for (; anIt != aLast; anIt++) {
191       aContext->Activate(anAIS, (*anIt));
192     }
193   }
194
195   if (isUpdateViewer)
196     updateViewer();
197 }
198
199 void XGUI_Displayer::deactivate(ObjectPtr theObject, bool toUpdate)
200 {
201   if (isVisible(theObject)) {
202     Handle(AIS_InteractiveContext) aContext = AISContext();
203
204      boost::shared_ptr<GeomAPI_AISObject> anObj = myResult2AISObjectMap[theObject];
205      Handle(AIS_InteractiveObject) anAIS = anObj->impl<Handle(AIS_InteractiveObject)>();
206      aContext->Deactivate(anAIS);
207   }
208 }
209
210 void XGUI_Displayer::stopSelection(const QList<ObjectPtr>& theResults, const bool isStop,
211                                    const bool isUpdateViewer)
212 {
213   Handle(AIS_InteractiveContext) aContext = AISContext();
214
215   Handle(AIS_Shape) anAIS;
216   QList<ObjectPtr>::const_iterator anIt = theResults.begin(), aLast = theResults.end();
217   ObjectPtr aFeature;
218   for (; anIt != aLast; anIt++) {
219     aFeature = *anIt;
220     if (isVisible(aFeature))
221       anAIS = Handle(AIS_Shape)::DownCast(myResult2AISObjectMap[aFeature]->impl<Handle(AIS_InteractiveObject)>());
222     if (anAIS.IsNull())
223       continue;
224
225     if (isStop) {
226       QColor aColor(Qt::white);
227       anAIS->SetColor(Quantity_Color(aColor.red()/255., aColor.green()/255., aColor.blue()/255., Quantity_TOC_RGB));
228       anAIS->Redisplay();
229     }
230     else {
231       QColor aColor(Qt::red);
232       anAIS->SetColor(Quantity_Color(aColor.red()/255., aColor.green()/255., aColor.blue()/255., Quantity_TOC_RGB));
233       anAIS->Redisplay();
234     }
235   }
236   if (isUpdateViewer)
237     updateViewer();
238 }
239
240 void XGUI_Displayer::setSelected(const QList<ObjectPtr>& theResults, const bool isUpdateViewer)
241 {
242   Handle(AIS_InteractiveContext) aContext = AISContext();
243   // we need to unhighligth objects manually in the current local context
244   // in couple with the selection clear (TODO)
245   Handle(AIS_LocalContext) aLocalContext = aContext->LocalContext();
246   if (!aLocalContext.IsNull())
247     aLocalContext->UnhilightLastDetected(myWorkshop->viewer()->activeView());
248
249   aContext->ClearSelected();
250   foreach(ObjectPtr aResult, theResults) {
251     if (myResult2AISObjectMap.find(aResult) == myResult2AISObjectMap.end()) 
252       continue;
253
254     boost::shared_ptr<GeomAPI_AISObject> anObj = myResult2AISObjectMap[aResult];
255     if (anObj) {
256       Handle(AIS_InteractiveObject) anAIS = anObj->impl<Handle(AIS_InteractiveObject)>();
257       if (!anAIS.IsNull())
258         aContext->AddOrRemoveSelected(anAIS, false);
259     }
260   }
261   if (isUpdateViewer)
262     updateViewer();
263 }
264
265
266 /*void XGUI_Displayer::EraseAll(const bool isUpdateViewer)
267 {
268   Handle(AIS_InteractiveContext) ic = AISContext();
269
270   AIS_ListOfInteractive aList;
271   ic->DisplayedObjects(aList);
272   AIS_ListIteratorOfListOfInteractive anIter(aList);
273   for (; anIter.More(); anIter.Next()) {
274     if ((anIter.Value()->DynamicType() == STANDARD_TYPE(AIS_Trihedron)))
275       continue;
276
277     // erase an object
278     Handle(AIS_InteractiveObject) anIO = anIter.Value();
279     ic->Erase(anIO, false);
280   }
281   myResult2AISObjectMap.clear();
282   if (isUpdateViewer)
283     updateViewer();
284 }*/
285
286 void XGUI_Displayer::eraseDeletedResults(const bool isUpdateViewer)
287 {
288   Handle(AIS_InteractiveContext) aContext = AISContext();
289
290   ResultToAISMap::const_iterator aFIt = myResult2AISObjectMap.begin(),
291                                  aFLast = myResult2AISObjectMap.end();
292   std::list<ObjectPtr> aRemoved;
293   for (; aFIt != aFLast; aFIt++)
294   {
295     ObjectPtr aFeature = (*aFIt).first;
296     if (!aFeature || !aFeature->data() || !aFeature->data()->isValid()) {
297       boost::shared_ptr<GeomAPI_AISObject> anObj = (*aFIt).second;
298       if (!anObj) continue;
299       Handle(AIS_InteractiveObject) anAIS = anObj->impl<Handle(AIS_InteractiveObject)>();
300       if (!anAIS.IsNull()) {
301         aContext->Erase(anAIS, false);
302         aRemoved.push_back(aFeature);
303       }
304     }
305   }
306   std::list<ObjectPtr>::const_iterator anIt = aRemoved.begin(),
307                                                                  aLast = aRemoved.end();
308   for (; anIt != aLast; anIt++) {
309     myResult2AISObjectMap.erase(myResult2AISObjectMap.find(*anIt));
310   }
311
312   if (isUpdateViewer)
313     updateViewer();
314 }
315
316 void XGUI_Displayer::closeLocalContexts(const bool isUpdateViewer)
317 {
318   closeAllContexts(true);
319 }
320
321 boost::shared_ptr<GeomAPI_AISObject> XGUI_Displayer::getAISObject(ObjectPtr theObject) const
322 {
323   boost::shared_ptr<GeomAPI_AISObject> anIO;
324   if (myResult2AISObjectMap.find(theObject) != myResult2AISObjectMap.end())
325     anIO = (myResult2AISObjectMap.find(theObject))->second;
326   return anIO;
327 }
328
329 ObjectPtr XGUI_Displayer::getObject(Handle(AIS_InteractiveObject) theIO) const
330 {
331   ObjectPtr aFeature;
332   ResultToAISMap::const_iterator aFIt = myResult2AISObjectMap.begin(),
333                                  aFLast = myResult2AISObjectMap.end();
334   for (; aFIt != aFLast && !aFeature; aFIt++) {
335     boost::shared_ptr<GeomAPI_AISObject> anObj = (*aFIt).second;
336     if (!anObj) continue;
337     Handle(AIS_InteractiveObject) anAIS = anObj->impl<Handle(AIS_InteractiveObject)>();
338     if (anAIS != theIO)
339       continue;
340     aFeature = (*aFIt).first;
341   }
342   return aFeature;
343 }
344
345 void XGUI_Displayer::closeAllContexts(const bool isUpdateViewer)
346 {
347   Handle(AIS_InteractiveContext) ic = AISContext();
348   if (!ic.IsNull()) {
349     ic->CloseAllContexts(false);
350     if (isUpdateViewer)
351       updateViewer();
352   }
353 }
354
355 void XGUI_Displayer::updateViewer()
356 {
357   Handle(AIS_InteractiveContext) ic = AISContext();
358   if (!ic.IsNull())
359     ic->UpdateCurrentViewer();
360 }
361
362 Handle(AIS_InteractiveContext) XGUI_Displayer::AISContext() const 
363
364   return myWorkshop->viewer()->AISContext(); 
365 }
366
367
368 void XGUI_Displayer::display(boost::shared_ptr<GeomAPI_AISObject> theAIS, bool isUpdate)
369 {
370   Handle(AIS_InteractiveContext) aContext = AISContext();
371   Handle(AIS_InteractiveObject) anAISIO = theAIS->impl<Handle(AIS_InteractiveObject)>();
372   if (!anAISIO.IsNull())
373     aContext->Display(anAISIO, isUpdate);
374 }
375
376 void XGUI_Displayer::erase(boost::shared_ptr<GeomAPI_AISObject> theAIS, const bool isUpdate)
377 {
378   Handle(AIS_InteractiveContext) aContext = AISContext();
379   Handle(AIS_InteractiveObject) anAISIO = theAIS->impl<Handle(AIS_InteractiveObject)>();
380   if (!anAISIO.IsNull()) {
381     aContext->Remove(anAISIO, isUpdate);
382   }
383 }
384