]> SALOME platform Git repositories - modules/shaper.git/blob - src/SHAPERGUI/SHAPERGUI.cpp
Salome HOME
Fix a bug with disabling of selection in another modules
[modules/shaper.git] / src / SHAPERGUI / SHAPERGUI.cpp
1 // Copyright (C) 2014-2019  CEA/DEN, EDF R&D
2 //
3 // This library is free software; you can redistribute it and/or
4 // modify it under the terms of the GNU Lesser General Public
5 // License as published by the Free Software Foundation; either
6 // version 2.1 of the License, or (at your option) any later version.
7 //
8 // This library is distributed in the hope that it will be useful,
9 // but WITHOUT ANY WARRANTY; without even the implied warranty of
10 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
11 // Lesser General Public License for more details.
12 //
13 // You should have received a copy of the GNU Lesser General Public
14 // License along with this library; if not, write to the Free Software
15 // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA
16 //
17 // See http://www.salome-platform.org/ or email : webmaster.salome@opencascade.com
18 //
19
20 #include "SHAPERGUI.h"
21 #include "SHAPERGUI_DataModel.h"
22 #include "SHAPERGUI_OCCSelector.h"
23 #include "SHAPERGUI_NestedButton.h"
24 #include "SHAPERGUI_ToolbarsMgr.h"
25
26 #include <XGUI_Workshop.h>
27 #include <XGUI_PropertyPanel.h>
28 #include <XGUI_ContextMenuMgr.h>
29 #include <XGUI_ObjectsBrowser.h>
30 #include <XGUI_OperationMgr.h>
31 #include <XGUI_Displayer.h>
32 #include <XGUI_MenuMgr.h>
33 #include <XGUI_FacesPanel.h>
34 #include <XGUI_SelectionActivate.h>
35 #include <XGUI_InspectionPanel.h>
36 #include <XGUI_ViewerProxy.h>
37
38 #include <ModuleBase_Operation.h>
39 #include <ModuleBase_Preferences.h>
40 #include <ModuleBase_ActionInfo.h>
41 #include <ModuleBase_IModule.h>
42
43 #include <ModelAPI_Tools.h>
44
45 #include <LightApp_Application.h>
46 #include <LightApp_SelectionMgr.h>
47 #include <LightApp_OCCSelector.h>
48 #include <LightApp_Study.h>
49
50 #include <OCCViewer_ViewModel.h>
51 #include <OCCViewer_ViewPort3d.h>
52
53 #include <SUIT_Selector.h>
54 #include <SUIT_Desktop.h>
55 #include <SUIT_ViewManager.h>
56 #include <SUIT_ViewWindow.h>
57 #include <SUIT_ResourceMgr.h>
58 #include <SUIT_DataBrowser.h>
59
60 #include <QtxPopupMgr.h>
61 #include <QtxActionMenuMgr.h>
62 #include <QtxActionToolMgr.h>
63 #include <QtxResourceMgr.h>
64
65 #include <Config_PropManager.h>
66 #include <Config_ModuleReader.h>
67
68 #include <AIS_ListOfInteractive.hxx>
69 #include <AIS_ListIteratorOfListOfInteractive.hxx>
70
71 #include <QDockWidget>
72 #include <QAction>
73 #include <QTimer>
74 #include <QMenu>
75 #include <QToolBar>
76
77 #define SALOME_PATCH_FOR_CTRL_WHEEL
78
79 extern "C" {
80 SHAPERGUI_EXPORT CAM_Module* createModule()
81 {
82   return new SHAPERGUI();
83 }
84
85 SHAPERGUI_EXPORT char* getModuleVersion()
86 {
87   return (char*)"0.0";
88 }
89 } // extern "C"
90
91
92 static const QString ToolbarsSection("SHAPER_Toolbars");
93 static const QString FreeCommandsParam("OutOFToolbars");
94
95
96 /** 
97 * Class for preferences management
98 */
99 class SHAPERGUI_PrefMgr: public ModuleBase_IPrefMgr
100 {
101 public:
102   /// Constructor
103   /// \param theMgr preferences manager of SALOME
104   /// \param theModName name of the module
105   SHAPERGUI_PrefMgr(LightApp_Preferences* theMgr, const QString& theModName):
106     myMgr(theMgr), myModName(theModName) {}
107
108   virtual int addPreference(const QString& theLbl, int pId,
109                             SUIT_PreferenceMgr::PrefItemType theType,
110                             const QString& theSection, const QString& theName )
111   {
112     return myMgr->addPreference(myModName, theLbl, pId, theType, theSection, theName);
113   }
114
115   virtual void setItemProperty(const QString& thePropName,
116                                const QVariant& theValue,
117                                const int theId = -1)
118   {
119     myMgr->setItemProperty(thePropName, theValue, theId);
120   }
121
122
123   virtual SUIT_PreferenceMgr* prefMgr() const { return myMgr; }
124
125 private:
126   LightApp_Preferences* myMgr;
127   QString myModName;
128 };
129
130
131
132
133 //******************************************************
134 SHAPERGUI::SHAPERGUI()
135     : LightApp_Module("SHAPER"),
136       mySelector(0), myIsOpened(0), myPopupMgr(0), myIsInspectionVisible(false),
137   myInspectionPanel(0), myIsToolbarsModified(false)
138 {
139   myWorkshop = new XGUI_Workshop(this);
140   connect(myWorkshop, SIGNAL(commandStatusUpdated()),
141           this, SLOT(onUpdateCommandStatus()));
142
143   myProxyViewer = new SHAPERGUI_SalomeViewer(this);
144
145   ModuleBase_Preferences::setResourceMgr(application()->resourceMgr());
146
147   // It will be called in XGUI_Workshop::startApplication
148   // ModuleBase_Preferences::loadCustomProps();
149 }
150
151 //******************************************************
152 SHAPERGUI::~SHAPERGUI()
153 {
154   delete myWorkshop;
155   delete myProxyViewer;
156 }
157
158 //******************************************************
159 void SHAPERGUI::initialize(CAM_Application* theApp)
160 {
161   LightApp_Module::initialize(theApp);
162
163   myWorkshop->startApplication();
164   LightApp_Application* anApp = dynamic_cast<LightApp_Application*>(theApp);
165   if (anApp)
166   {
167     connect(anApp, SIGNAL(preferenceResetToDefaults()), this, SLOT(onDefaultPreferences()));
168   }
169
170   int aMenu = createMenu(tr("Inspection"), -1, -1, 30);
171   int aSubMenu = createMenu(tr("Information"), aMenu);
172
173   int aId = getNextCommandId();
174   myActionsList.append(aId);
175   SUIT_Desktop* aDesk = application()->desktop();
176   QString aTip = tr("Show inspection window");
177   myWhatIsAction = createAction(aId, aTip, QIcon(":pictures/whatis.png"), tr("What Is"),
178     aTip, QKeySequence(), aDesk, true, this, SLOT(onWhatIs(bool)));
179   myWhatIsAction->setStatusTip(aTip);
180   myWhatIsAction->setData("INSPECTION_CMD");
181   createMenu(aId, aSubMenu, 0);
182
183   QString aToolName = tr("Inspection tool");
184   int aTool = createTool(aToolName);
185   int aToolId = createTool(myWhatIsAction, aTool);
186   registerCommandToolbar(aToolName, aId);
187
188   // Define Edit toolbars command
189   aId = getNextCommandId();
190   //myActionsList.append(aId); Do not use it for editing of toolbars
191   aTip = tr("Edit toolbars of the module");
192   QAction* aAction = createAction(aId, aTip, QIcon(":pictures/configure_toolbars.png"),
193     tr("Edit toolbars..."), aTip, QKeySequence(), aDesk, false, this, SLOT(onEditToolbars()));
194   int aEditMenu = createMenu(tr("MEN_DESK_EDIT"), -1, -1, 30);
195   int aEditItem = createMenu(aId, aEditMenu);
196
197   // Initialize viewer proxy if OCC viewer is already exist
198   ViewManagerList aOCCViewManagers;
199   application()->viewManagers(OCCViewer_Viewer::Type(), aOCCViewManagers);
200   if (aOCCViewManagers.size() > 0) {
201     SUIT_ViewManager* aMgr = aOCCViewManagers.first();
202     SUIT_ViewWindow* aWnd = aMgr->getActiveView();
203     if (aWnd) {
204       OCCViewer_ViewWindow* aOccWnd = static_cast<OCCViewer_ViewWindow*>(aWnd);
205       OCCViewer_ViewPort3d* aViewPort = aOccWnd->getViewPort();
206       if (aViewPort) {
207         XGUI_ViewerProxy* aViewer = myWorkshop->viewer();
208         aViewPort->installEventFilter(aViewer);
209         Handle(V3d_View) aView = aViewPort->getView();
210         aViewer->SetScale(aView, aView->Camera()->Scale());
211         // We can not create selector here because other modules will be deactivated later
212         //onViewManagerAdded(aMgr);
213       }
214     }
215   }
216 }
217
218 //******************************************************
219 void SHAPERGUI::windows(QMap<int, int>& theWndMap) const
220 {
221   theWndMap.insert(LightApp_Application::WT_PyConsole, Qt::BottomDockWidgetArea);
222 }
223
224 //******************************************************
225 void SHAPERGUI::viewManagers(QStringList& theList) const
226 {
227   theList.append(OCCViewer_Viewer::Type());
228 }
229
230 //******************************************************
231 // We can not create selector in this method because it can be called when
232 // SHAPER module is not active. Take into account that creation of our selector
233 // leads to switching OFF all other selectors
234 //void SHAPERGUI::connectToStudy(CAM_Study* theStudy)
235 //{
236 //  // if there are created viewer managers, we should try to create viewer
237 //  // selector and initialize viewer with it. It sets interactive context to the
238 //  // proxy viewer. If study is opened, CAM application calls this method before the open()
239 //  // of data model
240 //  // the SHAPER data model is specific and during open(load) redisplay signals are flushed, so
241 //  // we need to connect to the viewer before it. Here,
242 //  // it seems the most appropriate place for this
243 //  // according to SALOME architecture.
244 //  if (!mySelector) {
245 //    ViewManagerList OCCViewManagers;
246 //    application()->viewManagers(OCCViewer_Viewer::Type(), OCCViewManagers);
247 //    if (OCCViewManagers.size() > 0) {
248 //      mySelector = createSelector(OCCViewManagers.first());
249 //    }
250 //  }
251 //  LightApp_Module::connectToStudy(theStudy);
252 //}
253
254 //******************************************************
255 bool SHAPERGUI::activateModule(SUIT_Study* theStudy)
256 {
257   bool isDone = LightApp_Module::activateModule(theStudy);
258   loadToolbarsConfig();
259
260   SHAPERGUI_DataModel* aDataModel = dynamic_cast<SHAPERGUI_DataModel*>(dataModel());
261   aDataModel->initRootObject();
262
263   if (isDone) {
264     setMenuShown(true);
265     setToolShown(true);
266
267     QObject* aObj = myWorkshop->objectBrowser()->parent();
268     QDockWidget* aObjDoc = dynamic_cast<QDockWidget*>(aObj);
269     if (aObjDoc) {
270       QAction* aViewAct = aObjDoc->toggleViewAction();
271       aViewAct->setEnabled(true);
272       myWorkshop->objectBrowser()->setVisible(true);
273       aObjDoc->setVisible(true);
274       desktop()->tabifyDockWidget(aObjDoc, myWorkshop->propertyPanel());
275     }
276
277     if (!myInspectionPanel) {
278       myInspectionPanel = myWorkshop->inspectionPanel();
279       QAction* aViewAct = myInspectionPanel->toggleViewAction();
280       connect(aViewAct, SIGNAL(toggled(bool)), this, SLOT(onWhatIs(bool)));
281     }
282     myInspectionPanel->toggleViewAction()->setEnabled(true);
283
284     if (!mySelector) {
285       ViewManagerList OCCViewManagers;
286       application()->viewManagers(OCCViewer_Viewer::Type(), OCCViewManagers);
287       if (OCCViewManagers.size() > 0) {
288         onViewManagerAdded(OCCViewManagers.first());
289       }
290     }
291     // it should be performed after the selector creation in order to have AISContext
292     myWorkshop->activateModule();
293     //action(myEraseAll)->setEnabled(false);
294
295     if (myIsOpened) {
296       myWorkshop->objectBrowser()->rebuildDataTree();
297       myWorkshop->updateCommandStatus();
298       myIsOpened = false;
299     }
300     else
301       myWorkshop->updateCommandStatus();
302   }
303   SUIT_ResourceMgr* aResMgr = application()->resourceMgr();
304   myIsStorePositions = aResMgr->booleanValue("Study", "store_positions", true);
305   myIsEditEnabled = getApp()->isEditEnabled();
306   getApp()->setEditEnabled(false);
307
308   // this following row is caused by #187 bug.
309   // SALOME saves the dock widget positions before deactivateModule() and
310   // load it after the module activation. So, if the panel is visible before
311   // deactivate, it becomes visible after activate.
312   // In order to avoid the visible property panel, the widget position save is
313   // switch off in this module
314   aResMgr->setValue("Study", "store_positions", false);
315
316   // Synchronize displayed objects
317   Handle(AIS_InteractiveContext) aContext;
318   if (mySelector && mySelector->viewer())
319     aContext = mySelector->viewer()->getAISContext();
320
321   if (!aContext.IsNull()) {
322     XGUI_Displayer* aDisp = myWorkshop->displayer();
323     QObjectPtrList aObjList = aDisp->displayedObjects();
324
325     //if (myHighlightPointAspect.IsNull()) {
326     //  Handle(AIS_Trihedron) aTrihedron = mySelector->viewer()->getTrihedron();
327     //  myHighlightPointAspect =
328     //    new Graphic3d_AspectMarker3d(aTrihedron->getHighlightPointAspect()->Aspect().operator*());
329     //}
330     if (myOldSelectionColor.size() == 0)
331       myOldSelectionColor = aDisp->selectionColor();
332
333     AIS_ListOfInteractive aList;
334     aContext->DisplayedObjects(aList);
335     AIS_ListIteratorOfListOfInteractive aLIt;
336     Handle(AIS_InteractiveObject) anAISIO;
337     foreach (ObjectPtr aObj, aObjList) {
338       AISObjectPtr aPrs = aDisp->getAISObject(aObj);
339       Handle(AIS_InteractiveObject) aAIS = aPrs->impl<Handle(AIS_InteractiveObject)>();
340       bool aFound = false;
341       for (aLIt.Initialize(aList); aLIt.More(); aLIt.Next()) {
342         anAISIO = aLIt.Value();
343         if (anAISIO.get() == aAIS.get()) {
344           aFound = true;
345           break;
346         }
347       }
348       if (!aFound) {
349         aObj->setDisplayed(false);
350         //aDisp->erase(aObj, false);
351       }
352     }
353     Events_Loop::loop()->flush(Events_Loop::eventByName(EVENT_OBJECT_TO_REDISPLAY));
354   }
355   myProxyViewer->activateViewer(true);
356
357   // Post-processing for LoadScriptId to remove created(if it was created) SALOME Object Browser
358   connect(getApp()->action(LightApp_Application::UserID+1), SIGNAL(triggered(bool)),
359           this, SLOT(onScriptLoaded()));
360
361   disconnect(getApp()->action(LightApp_Application::FileSaveId), SIGNAL(triggered(bool)),
362              getApp(), SLOT(onSaveDoc()));
363   disconnect(getApp()->action(LightApp_Application::FileSaveAsId), SIGNAL(triggered(bool)),
364              getApp(), SLOT(onSaveAsDoc()));
365
366   connect(getApp()->action(LightApp_Application::FileSaveId), SIGNAL(triggered(bool)),
367           this, SLOT(onSaveDocByShaper()));
368   connect(getApp()->action(LightApp_Application::FileSaveAsId), SIGNAL(triggered(bool)),
369           this, SLOT(onSaveAsDocByShaper()));
370
371   return isDone;
372 }
373
374 //******************************************************
375 bool SHAPERGUI::deactivateModule(SUIT_Study* theStudy)
376 {
377   saveToolbarsConfig();
378
379   myProxyViewer->activateViewer(false);
380   setMenuShown(false);
381   setToolShown(false);
382
383   myWorkshop->deactivateModule();
384
385   QObject* aObj = myWorkshop->objectBrowser()->parent();
386   QDockWidget* aObjDoc = dynamic_cast<QDockWidget*>(aObj);
387   if (aObjDoc) {
388     aObjDoc->setVisible(false);
389     myWorkshop->objectBrowser()->setVisible(false);
390     QAction* aViewAct = aObjDoc->toggleViewAction();
391     aViewAct->setEnabled(false);
392   }
393
394   myIsInspectionVisible = myInspectionPanel->isVisible();
395   myInspectionPanel->hide();
396   QAction* aViewAct = myInspectionPanel->toggleViewAction();
397   aViewAct->setEnabled(false);
398
399   // the active operation should be stopped for the next activation.
400   // There should not be active operation and visualized preview.
401   // Abort operation should be performed before the selection's remove
402   // because the displayed objects should be removed from the viewer, but
403   // the AIS context is obtained from the selector.
404   ModuleBase_Operation* anOperation = myWorkshop->operationMgr()->currentOperation();
405   while (anOperation) {
406     anOperation->abort();
407     anOperation = myWorkshop->operationMgr()->currentOperation();
408   }
409   // Delete selector because it has to be redefined on next activation
410   if (mySelector) {
411     //if (!myHighlightPointAspect.IsNull()) {
412     //  Handle(AIS_Trihedron) aTrihedron = mySelector->viewer()->getTrihedron();
413     //  aTrihedron->getHighlightPointAspect()->SetAspect(myHighlightPointAspect);
414     //  myHighlightPointAspect.Nullify();
415     //}
416     myWorkshop->displayer()->setSelectionColor(myOldSelectionColor);
417     myProxyViewer->setSelector(0);
418
419     LightApp_SelectionMgr* aMgr = getApp()->selectionMgr();
420     QList<SUIT_Selector*> aList;
421     aMgr->selectors(aList);
422     foreach(SUIT_Selector* aSel, aList) {
423       aSel->setEnabled(aSel != mySelector);
424     }
425
426     delete mySelector;
427     mySelector = 0;
428   }
429
430   myWorkshop->hidePanel(myWorkshop->facesPanel());
431   //myWorkshop->contextMenuMgr()->disconnectViewer();
432
433   SUIT_ResourceMgr* aResMgr = application()->resourceMgr();
434   aResMgr->setValue("Study", "store_positions", myIsStorePositions);
435   getApp()->setEditEnabled(myIsEditEnabled);
436
437   myOldSelectionColor.clear();
438
439   // Post-processing for LoadScriptId to remove created(if it was created) SALOME Object Browser
440   disconnect(getApp()->action(LightApp_Application::UserID+1), SIGNAL(triggered(bool)),
441              this, SLOT(onScriptLoaded()));
442
443   disconnect(getApp()->action(LightApp_Application::FileSaveId), SIGNAL(triggered(bool)),
444              this, SLOT(onSaveDocByShaper()));
445   disconnect(getApp()->action(LightApp_Application::FileSaveAsId), SIGNAL(triggered(bool)),
446              this, SLOT(onSaveAsDocByShaper()));
447
448   connect(getApp()->action(LightApp_Application::FileSaveId), SIGNAL(triggered(bool)),
449           getApp(), SLOT(onSaveDoc()));
450   connect(getApp()->action(LightApp_Application::FileSaveAsId), SIGNAL(triggered(bool)),
451           getApp(), SLOT(onSaveAsDoc()));
452
453   publishToStudy();
454
455   return LightApp_Module::deactivateModule(theStudy);
456 }
457
458 //******************************************************
459 void SHAPERGUI::onViewManagerAdded(SUIT_ViewManager* theMgr)
460 {
461   if (!mySelector) {
462     mySelector = createSelector(theMgr);
463     myWorkshop->selectionActivate()->updateSelectionFilters();
464     myWorkshop->selectionActivate()->updateSelectionModes();
465     myWorkshop->synchronizeViewer();
466   }
467 }
468
469 //******************************************************
470 void SHAPERGUI::onViewManagerRemoved(SUIT_ViewManager* theMgr)
471 {
472   if (mySelector) {
473     if (theMgr->getType() == OCCViewer_Viewer::Type()) {
474       OCCViewer_Viewer* aViewer = static_cast<OCCViewer_Viewer*>(theMgr->getViewModel());
475       if (mySelector->viewer() == aViewer) {
476         XGUI_Displayer* aDisp = myWorkshop->displayer();
477         QObjectPtrList aObjects = aDisp->displayedObjects();
478         ResultPtr aRes;
479         foreach(ObjectPtr aObj, aObjects) {
480           aObj->setDisplayed(false);
481           aRes = std::dynamic_pointer_cast<ModelAPI_Result>(aObj);
482           if (aRes.get()) {
483             while (aRes = ModelAPI_Tools::bodyOwner(aRes)) {
484               aRes->setDisplayed(false);
485             }
486           }
487         }
488         Events_Loop::loop()->flush(Events_Loop::eventByName(EVENT_OBJECT_TO_REDISPLAY));
489         myProxyViewer->setSelector(0);
490         delete mySelector;
491         mySelector = 0;
492
493         myWorkshop->module()->clearViewer();
494       }
495     }
496   }
497 }
498
499 //******************************************************
500 QtxPopupMgr* SHAPERGUI::popupMgr()
501 {
502   if (!myPopupMgr)
503     myPopupMgr = new QtxPopupMgr( 0, this );
504   return myPopupMgr;
505 }
506
507 //******************************************************
508 void SHAPERGUI::onDefaultPreferences()
509 {
510   // reset main resources
511   ModuleBase_Preferences::resetResourcePreferences(preferences());
512   // reset plugin's resources
513   ModuleBase_Preferences::resetConfigPropPreferences(preferences());
514
515   myWorkshop->displayer()->redisplayObjects();
516 }
517
518 //******************************************************
519 void SHAPERGUI::onScriptLoaded()
520 {
521   // this slot is called after processing of the LoadScriptId action of SalomeApp Application
522   // Each dumped script contains updateObjBrowser() that creates a new instance of Object
523   // Browser. When SHAPER module is active, this browser should not be used. It might be removed
524   // as hidden by means of updateWindows() of SalomeApp_Application or to remove
525   // it manually (because this method of application is protected)
526   SUIT_DataBrowser* aBrowser = getApp()->objectBrowser();
527   if (aBrowser)
528     delete aBrowser;
529   myWorkshop->displayer()->updateViewer();
530   myWorkshop->updateCommandStatus();
531 }
532
533 //******************************************************
534 void SHAPERGUI::onSaveDocByShaper()
535 {
536   if(!workshop()->operationMgr()->abortAllOperations(XGUI_OperationMgr::XGUI_InformationMessage))
537     return;
538
539   getApp()->onSaveDoc();
540 }
541
542 //******************************************************
543 void SHAPERGUI::onSaveAsDocByShaper()
544 {
545   if(!workshop()->operationMgr()->abortAllOperations(XGUI_OperationMgr::XGUI_InformationMessage))
546     return;
547
548   getApp()->onSaveAsDoc();
549 }
550
551 //******************************************************
552 void SHAPERGUI::onUpdateCommandStatus()
553 {
554   getApp()->updateActions();
555 }
556
557 //******************************************************
558 SHAPERGUI_OCCSelector* SHAPERGUI::createSelector(SUIT_ViewManager* theMgr)
559 {
560   if (theMgr->getType() == OCCViewer_Viewer::Type()) {
561     OCCViewer_Viewer* aViewer = static_cast<OCCViewer_Viewer*>(theMgr->getViewModel());
562
563     //if (myHighlightPointAspect.IsNull()) {
564     //  Handle(AIS_Trihedron) aTrihedron = aViewer->getTrihedron();
565     //  myHighlightPointAspect =
566     //    new Graphic3d_AspectMarker3d(aTrihedron->getHighlightPointAspect()->Aspect().operator*());
567     //}
568     SHAPERGUI_OCCSelector* aSelector = new SHAPERGUI_OCCSelector(aViewer,
569                                                                  getApp()->selectionMgr());
570 #ifdef SALOME_PATCH_FOR_CTRL_WHEEL
571     aViewer->setUseLocalSelection(true);
572 #endif
573     LightApp_SelectionMgr* aMgr = getApp()->selectionMgr();
574     QList<SUIT_Selector*> aList;
575     aMgr->selectors(aList);
576     foreach(SUIT_Selector* aSel, aList)
577     {
578       aSel->setEnabled(aSel == aSelector);
579     }
580     myProxyViewer->setSelector(aSelector);
581
582     if (myOldSelectionColor.size() == 0)
583       myOldSelectionColor = myWorkshop->displayer()->selectionColor();
584
585     std::vector<int> aColor = Config_PropManager::color("Visualization", "selection_color");
586     myWorkshop->displayer()->setSelectionColor(aColor);
587     return aSelector;
588   }
589   return 0;
590 }
591
592 //******************************************************
593 CAM_DataModel* SHAPERGUI::createDataModel()
594 {
595   return new SHAPERGUI_DataModel(this);
596 }
597
598 QAction* SHAPERGUI::addFeature(const QString& theWBName, const ActionInfo& theInfo,
599                                const bool isAddSeparator)
600 {
601   return addFeature(theWBName,
602                     theInfo.toolBar,
603                     theInfo.id,
604                     theInfo.text,
605                     //Issue #650: in the SALOME mode the tooltip should be same as text
606                     theInfo.text,
607                     theInfo.icon,
608                     theInfo.shortcut,
609                     theInfo.checkable,
610                     isAddSeparator,
611                     theInfo.toolTip);
612 }
613
614 //******************************************************
615 QAction* SHAPERGUI::addFeature(const QString& theWBName, const QString& theTBName,
616                                const QString& theId, const QString& theTitle, const QString& theTip,
617                                const QIcon& theIcon, const QKeySequence& theKeys,
618                                bool isCheckable, const bool isAddSeparator,
619                                const QString& theStatusTip)
620 {
621   static QString aLastTool = "";
622   static int aNb = 0;
623   if (aLastTool.isEmpty())
624     aLastTool = theWBName;
625   else if (theWBName != aLastTool) {
626     aLastTool = theWBName;
627     if (aNb > 20) {
628       desktop()->addToolBarBreak();
629       aNb = 0;
630     }
631   }
632   aNb++;
633
634   int aId = getNextCommandId();
635   myActionsList.append(aId);
636   SUIT_Desktop* aDesk = application()->desktop();
637   int aKeys = 0;
638   for (int i = 0; i < theKeys.count(); i++)
639     aKeys += theKeys[i];
640   QAction* aAction = createAction(aId, theTip, theIcon, theTitle, theTip, aKeys, aDesk,
641                                   isCheckable);
642   aAction->setStatusTip(theStatusTip);
643
644   aAction->setData(theId);
645
646   int aWBMenu = createMenu(theWBName, -1, -1, 30/*10-Window, 1000 - Help*/);
647   int aItemId = createMenu(aId, aWBMenu);
648   if (isAddSeparator)
649     createMenu(separator(), aWBMenu);
650
651   int aWBTool = createTool(theTBName, theTBName);
652   int aToolId = createTool(aId, aWBTool);
653   registerCommandToolbar(theTBName, aId);
654   if (isAddSeparator) {
655     createTool(separator(), aWBTool);
656     registerCommandToolbar(theTBName, -1);
657   }
658   return aAction;
659 }
660
661 bool SHAPERGUI::isFeatureOfNested(const QAction* theAction)
662 {
663   return dynamic_cast<const SHAPERGUI_NestedButton*>(theAction);
664 }
665
666 QAction* SHAPERGUI::addFeatureOfNested(const QString& theWBName,
667                                        const ActionInfo& theInfo,
668                                        const QList<QAction*>& theNestedActions)
669 {
670   SUIT_Desktop* aDesk = application()->desktop();
671   SHAPERGUI_NestedButton* anAction = new SHAPERGUI_NestedButton(aDesk, theNestedActions);
672   anAction->setData(theInfo.id);
673   anAction->setCheckable(theInfo.checkable);
674   anAction->setChecked(theInfo.checked);
675   anAction->setEnabled(theInfo.enabled);
676   anAction->setVisible(theInfo.visible);
677   anAction->setIcon(theInfo.icon);
678   anAction->setText(theInfo.text);
679   anAction->setToolTip(theInfo.toolTip);
680   anAction->setShortcut(theInfo.shortcut);
681   anAction->setFont(theInfo.font);
682
683   int aWBMenu = createMenu(theWBName, -1, -1, 30);
684   int aItemId = createMenu(anAction, aWBMenu);
685   myActionsList.append(aItemId);
686   createMenu(separator(), aWBMenu); /// nested action is always separated of others
687
688   int aWBTool = createTool(theWBName, theWBName);
689   int aToolId = createTool(anAction, aWBTool);
690   registerCommandToolbar(theWBName, aItemId);
691   createTool(separator(), aWBTool); /// nested action is always separated of others
692   registerCommandToolbar(theWBName, -1);
693
694   return anAction;
695 }
696
697
698 //******************************************************
699 QAction* SHAPERGUI::addDesktopCommand(const QString& theId, const QString& theTitle,
700                                            const QString& theTip, const QIcon& theIcon,
701                                            const QKeySequence& theKeys, bool isCheckable,
702                                            const char* theMenuSourceText, const int theMenuPosition)
703 {
704   int aMenu = createMenu(tr(theMenuSourceText), -1, -1);
705
706   int aId = getNextCommandId();
707   myActionsList.append(aId);
708   SUIT_Desktop* aDesk = application()->desktop();
709   int aKeys = 0;
710   for (int i = 0; i < theKeys.count(); i++)
711     aKeys += theKeys[i];
712   QAction* aAction = createAction(aId, theTip, theIcon, theTitle, theTip, aKeys, aDesk,
713                                   isCheckable);
714   aAction->setStatusTip(theTip);
715   aAction->setData(theId);
716   createMenu(aId, aMenu, theMenuPosition);
717   return aAction;
718 }
719
720 //******************************************************
721 void SHAPERGUI::addDesktopMenuSeparator(const char* theMenuSourceText, const int theMenuPosition)
722 {
723   int aMenu = createMenu(tr(theMenuSourceText), -1, -1);
724   createMenu(separator(), aMenu, -1, theMenuPosition);
725 }
726
727 //******************************************************
728 bool SHAPERGUI::addActionInToolbar( QAction* theAction, const QString& theToolBarTitle )
729 {
730   if( !theAction )
731     return false;
732
733   SUIT_Desktop* aDesktop = application()->desktop();
734   if( !aDesktop )
735     return false;
736
737   QtxActionToolMgr* aToolMgr = aDesktop->toolMgr();
738   if( !aToolMgr )
739     return false;
740
741   aToolMgr->append( theAction, theToolBarTitle );
742   return true;
743 }
744
745 //******************************************************
746 QList<QAction*> SHAPERGUI::commandList() const
747 {
748   QList<QAction*> aActions;
749   foreach (int aId, myActionsList) {
750     QAction* aCmd = action(aId);
751     if (aCmd)
752       aActions.append(aCmd);
753   }
754
755   return aActions;
756 }
757
758 //******************************************************
759 QMainWindow* SHAPERGUI::desktop() const
760 {
761   return application()->desktop();
762 }
763
764 void SHAPERGUI::setFeatureInfo(const QString& theFeatureId,
765                                const std::shared_ptr<Config_FeatureMessage>& theMessage)
766 {
767   myFeaturesInfo.insert(theFeatureId, theMessage);
768 }
769
770 std::shared_ptr<Config_FeatureMessage> SHAPERGUI::featureInfo(const QString& theFeatureId)
771 {
772   std::shared_ptr<Config_FeatureMessage> aMessage;
773   if (myFeaturesInfo.contains(theFeatureId))
774     aMessage =  myFeaturesInfo[theFeatureId];
775   return aMessage;
776 }
777
778 //******************************************************
779 void SHAPERGUI::selectionChanged()
780 {
781   LightApp_Module::selectionChanged();
782   myWorkshop->salomeViewerSelectionChanged();
783 }
784
785 //******************************************************
786 void SHAPERGUI::contextMenuPopup(const QString& theClient, QMenu* theMenu, QString& theTitle)
787 {
788   myWorkshop->contextMenuMgr()->updateViewerMenu();
789   myWorkshop->contextMenuMgr()->addViewerMenu(theMenu);
790   LightApp_Module::contextMenuPopup(theClient, theMenu, theTitle);
791 }
792
793
794 //******************************************************
795 void SHAPERGUI::createPreferences()
796 {
797   LightApp_Preferences* pref = preferences();
798   if (!pref)
799     return;
800   ModuleBase_Preferences::updateConfigByResources();
801   QString aModName = moduleName();
802
803   QtxPreferenceItem* item = pref->findItem(aModName, true );
804   if ( item && (!item->isEmpty() )) {
805     item->parentItem()->removeItem(item);
806     delete item;
807   }
808
809   int catId = pref->addPreference(aModName, -1 );
810   if ( catId == -1 )
811     return;
812   SHAPERGUI_PrefMgr aMgr(pref, aModName);
813   ModuleBase_Preferences::createEditContent(&aMgr, catId);
814
815   int viewTab = pref->addItem(tr("Viewer"), catId);
816   // Create other parameters group in viewer tab
817   int otherGroup = pref->addItem(tr("Default selection"), viewTab);
818   pref->setItemProperty("columns", 3, otherGroup);
819   pref->addItem(tr("Faces"), otherGroup,
820                          SUIT_PreferenceMgr::Bool,
821                          ModuleBase_Preferences::VIEWER_SECTION, "face-selection");
822   pref->addItem(tr("Edges"), otherGroup,
823                          SUIT_PreferenceMgr::Bool,
824                          ModuleBase_Preferences::VIEWER_SECTION, "edge-selection");
825   pref->addItem(tr("Vertices"), otherGroup,
826                          SUIT_PreferenceMgr::Bool,
827                          ModuleBase_Preferences::VIEWER_SECTION, "vertex-selection");
828
829   int sensitivityGroup = pref->addItem(tr("Selection sensitivity"), viewTab);
830   pref->setItemProperty("columns", 2, sensitivityGroup);
831   pref->addItem(tr("Vertex"), sensitivityGroup, SUIT_PreferenceMgr::DblSpin,
832                 ModuleBase_Preferences::VIEWER_SECTION, "point-selection-sensitivity");
833   pref->addItem(tr("Edge"), sensitivityGroup, SUIT_PreferenceMgr::DblSpin,
834                 ModuleBase_Preferences::VIEWER_SECTION, "edge-selection-sensitivity");
835
836   int highlightGroup = pref->addItem(tr("Additional highlighting"), viewTab);
837   pref->setItemProperty("columns", 2, highlightGroup);
838   pref->addItem(tr("In 3d mode"), highlightGroup,
839     SUIT_PreferenceMgr::Bool, ModuleBase_Preferences::VIEWER_SECTION, "highlighting-3d");
840   pref->addItem(tr("In 2d mode"), highlightGroup,
841     SUIT_PreferenceMgr::Bool, ModuleBase_Preferences::VIEWER_SECTION, "highlighting-2d");
842
843   int colorScaleGroup = pref->addItem(tr("Color scale"), viewTab);
844   pref->setItemProperty("columns", 4, colorScaleGroup);
845   int aItem = aMgr.addPreference(tr("X position"), colorScaleGroup,
846     SUIT_PreferenceMgr::Double, ModuleBase_Preferences::VIEWER_SECTION, "scalar_bar_x_position");
847   pref->setItemProperty("min", 0, aItem);
848   pref->setItemProperty("max", 1, aItem);
849
850   aItem = aMgr.addPreference(tr("Y position"), colorScaleGroup,
851     SUIT_PreferenceMgr::Double, ModuleBase_Preferences::VIEWER_SECTION, "scalar_bar_y_position");
852   pref->setItemProperty("min", 0, aItem);
853   pref->setItemProperty("max", 1, aItem);
854
855   aItem = aMgr.addPreference(tr("Width"), colorScaleGroup,
856     SUIT_PreferenceMgr::Double, ModuleBase_Preferences::VIEWER_SECTION, "scalar_bar_width");
857   pref->setItemProperty("min", 0, aItem);
858   pref->setItemProperty("max", 1, aItem);
859
860   aItem = aMgr.addPreference(tr("Height"), colorScaleGroup,
861     SUIT_PreferenceMgr::Double, ModuleBase_Preferences::VIEWER_SECTION, "scalar_bar_height");
862   pref->setItemProperty("min", 0, aItem);
863   pref->setItemProperty("max", 1, aItem);
864
865   aItem = aMgr.addPreference(tr("Intervals number"), colorScaleGroup,
866     SUIT_PreferenceMgr::Integer, ModuleBase_Preferences::VIEWER_SECTION, "scalar_bar_nb_intervals");
867   pref->setItemProperty("min", 0, aItem);
868   pref->setItemProperty("max", 100, aItem);
869
870   aItem = aMgr.addPreference(tr("Text height"), colorScaleGroup,
871     SUIT_PreferenceMgr::Integer, ModuleBase_Preferences::VIEWER_SECTION, "scalar_bar_text_height");
872   pref->setItemProperty("min", 0, aItem);
873   pref->setItemProperty("max", 100, aItem);
874
875   aItem = aMgr.addPreference(tr("Text color"), colorScaleGroup,
876     SUIT_PreferenceMgr::Color, ModuleBase_Preferences::VIEWER_SECTION, "scalar_bar_text_color");
877
878   pref->retrieve();
879 }
880
881 //******************************************************
882 void SHAPERGUI::preferencesChanged(const QString& theSection, const QString& theParam)
883 {
884   SUIT_ResourceMgr* aResMgr = application()->resourceMgr();
885   QString aVal = aResMgr->stringValue(theSection, theParam);
886   Config_Prop* aProp = Config_PropManager::findProp(theSection.toStdString(),
887                                                     theParam.toStdString());
888   std::string aValue = aVal.toStdString();
889   if (aValue.empty()) {
890     aValue = aProp->defaultValue();
891     aResMgr->setValue(theSection, theParam, QString(aValue.c_str()));
892
893     LightApp_Preferences* pref = preferences();
894     if (pref)
895       pref->retrieve();
896   }
897   aProp->setValue(aValue);
898
899   if ((theSection == "Visualization") && (theParam == "selection_color")) {
900     std::vector<int> aColor = Config_PropManager::color("Visualization", "selection_color");
901     myWorkshop->displayer()->setSelectionColor(aColor);
902   }
903
904   myWorkshop->displayer()->redisplayObjects();
905 }
906
907 void SHAPERGUI::putInfo(const QString& theInfo, const int theMSecs)
908 {
909   application()->putInfo(theInfo, theMSecs);
910 }
911
912 bool SHAPERGUI::abortAllOperations()
913 {
914   return workshop()->operationMgr()->abortAllOperations();
915 }
916
917 void SHAPERGUI::createFeatureActions()
918 {
919   myWorkshop->menuMgr()->createFeatureActions();
920 }
921
922 void SHAPERGUI::onWhatIs(bool isToggled)
923 {
924   if (sender() == myWhatIsAction) {
925     QAction* aViewAct = myInspectionPanel->toggleViewAction();
926     aViewAct->blockSignals(true);
927     aViewAct->setChecked(isToggled);
928     aViewAct->blockSignals(false);
929     myInspectionPanel->setVisible(isToggled);
930   }
931   else {
932     myWhatIsAction->blockSignals(true);
933     myWhatIsAction->setChecked(isToggled);
934     myWhatIsAction->blockSignals(false);
935     myInspectionPanel->setVisible(isToggled);
936   }
937 }
938
939 void SHAPERGUI::updateModuleVisibilityState()
940 {
941   LightApp_Module::updateModuleVisibilityState();
942   onWhatIs(myIsInspectionVisible);
943 }
944
945 void SHAPERGUI::onEditToolbars()
946 {
947   SHAPERGUI_ToolbarsDlg aDlg(this);
948   if (aDlg.exec() == QDialog::Accepted) {
949     if (aDlg.isReset())
950       resetToolbars();
951     else
952       updateToolbars(aDlg.result());
953   }
954 }
955
956 void SHAPERGUI::registerCommandToolbar(const QString& theToolName, int theCommandId)
957 {
958   if (!myToolbars.contains(theToolName))
959     myToolbars[theToolName] = QList<int>();
960   myToolbars[theToolName].append(theCommandId);
961 }
962
963 int SHAPERGUI::getNextCommandId() const
964 {
965   QtxActionMenuMgr* aMenuMgr = menuMgr();
966   QIntList aIds = aMenuMgr->idList();
967   int aId = aIds.count();
968   while (action(aId) || myActionsList.contains(aId))
969     aId++;
970   return aId;
971 }
972
973 void SHAPERGUI::updateToolbars(const QMap<QString, QIntList>& theNewToolbars)
974 {
975   // Store default toolbars
976   if (myDefaultToolbars.size() == 0)
977     myDefaultToolbars = myToolbars;
978
979   QtxActionToolMgr* aMgr = toolMgr();
980   QStringList aToolbars = theNewToolbars.keys();
981   QIntList aCommands, aOldCmd;
982   int aToolbarId;
983   QAction* aAction;
984   int aActionId;
985   foreach(QString aName, aToolbars) {
986     aCommands = theNewToolbars[aName];
987     // Find or create toolbar
988     if (aMgr->hasToolBar(aName)) {
989       aToolbarId = aMgr->find(aMgr->toolBar(aName));
990       aOldCmd = myToolbars[aName];
991     }
992     else {
993       aToolbarId = aMgr->createToolBar(aName);
994     }
995     int aPos = 0;
996     foreach(int aCmd, aCommands) {
997       // Find action
998       if (aCmd == -1)
999         aAction = separator();
1000       else
1001         aAction = action(aCmd);
1002       aActionId = aMgr->actionId(aAction);
1003       if (aActionId == -1) {
1004         // Add new action
1005         aMgr->insert(aAction, aToolbarId, aPos);
1006       }
1007       else {
1008         // Change position of action
1009         if (aMgr->index(aActionId, aToolbarId) != aPos) {
1010           if (aMgr->containsAction(aActionId, aToolbarId))
1011             aMgr->remove(aActionId, aToolbarId);
1012           aMgr->insert(aActionId, aToolbarId, aPos);
1013         }
1014       }
1015       aOldCmd.removeAll(aCmd);
1016       aPos++;
1017     }
1018     // remove extra actions
1019     foreach(int aCmd, aOldCmd) {
1020       aAction = action(aCmd);
1021       aActionId = aMgr->actionId(aAction);
1022       aMgr->remove(aActionId, aToolbarId);
1023     }
1024     myToolbars.remove(aName);
1025   }
1026   // Remove extra toolbars
1027   aToolbars = myToolbars.keys();
1028   foreach(QString aName, aToolbars) {
1029     aMgr->removeToolBar(aName);
1030   }
1031   // Set new toolbars structure
1032   myToolbars = theNewToolbars;
1033   myIsToolbarsModified = true;
1034 }
1035
1036 void SHAPERGUI::saveToolbarsConfig()
1037 {
1038   if (!myIsToolbarsModified)
1039     return;
1040   // Save toolbars configuration into map
1041   QMap<QString, QStringList> aToolbarsConfig;
1042   QtxActionToolMgr* aMgr = toolMgr();
1043   QStringList aToolbars = myToolbars.keys();
1044   QIntList aActionsIds;
1045   foreach(QString aName, aToolbars) {
1046     aActionsIds = myToolbars[aName];
1047     QStringList aContent;
1048     foreach(int aId, aActionsIds) {
1049       if (aId == -1)
1050         aContent.append("");
1051       else
1052         aContent.append(action(aId)->data().toString());
1053     }
1054     aToolbarsConfig[aName] = aContent;
1055   }
1056   // Store the configuration into resources
1057   SUIT_ResourceMgr* aResMgr = application()->resourceMgr();
1058   QStringList aNames = aToolbarsConfig.keys();
1059   QStringList aValues;
1060   foreach(QString aToolbar, aNames) {
1061     aResMgr->setValue(ToolbarsSection, aToolbar, aToolbarsConfig[aToolbar].join(","));
1062   }
1063   // Remove obsolete parameters from resources
1064   QStringList aOldParams = aResMgr->parameters(ToolbarsSection);
1065   foreach(QString aName, aOldParams) {
1066     if (!aToolbars.contains(aName))
1067       aResMgr->remove(ToolbarsSection, aName);
1068   }
1069   // Store current list of free commands
1070   QIntList aFreeCommands = getFreeCommands();
1071   QStringList aFreeList;
1072   foreach(int aId, aFreeCommands) {
1073     aFreeList.append(action(aId)->data().toString());
1074   }
1075   if (aFreeList.size() > 0)
1076     aResMgr->setValue(ToolbarsSection, FreeCommandsParam, aFreeList.join(","));
1077
1078   myIsToolbarsModified = false;
1079 }
1080
1081 void SHAPERGUI::loadToolbarsConfig()
1082 {
1083   SUIT_ResourceMgr* aResMgr = application()->resourceMgr();
1084   QStringList aToolbarNames = aResMgr->parameters(ToolbarsSection);
1085   if (aToolbarNames.size() == 0)
1086     return;
1087
1088   // Create commands map
1089   QMap<QString, int> aCommandsMap;
1090   QString aCmdIdStr;
1091   foreach(int aId, myActionsList) {
1092     aCmdIdStr = action(aId)->data().toString();
1093     aCommandsMap[aCmdIdStr] = aId;
1094   }
1095
1096   // Create new toolbars structure
1097   QMap<QString, QIntList> aToolbars;
1098   QStringList aCommands;
1099   QIntList aKnownCommands;
1100   QList<QAction*> aActions;
1101   foreach(QString aName, aToolbarNames) {
1102     aCommands = aResMgr->stringValue(ToolbarsSection, aName).split(",");
1103     if (aName == FreeCommandsParam) {
1104       // The value is a list of free commands
1105       foreach(QString aCommand, aCommands) {
1106         aKnownCommands.append(aCommandsMap[aCommand]);
1107       }
1108     }
1109     else {
1110       aToolbars[aName] = QIntList();
1111       if (aCommands.size() > 0) {
1112         foreach(QString aCommand, aCommands) {
1113           if (aCommand.isEmpty())
1114             aToolbars[aName].append(-1);
1115           else if (aCommandsMap.contains(aCommand)) {
1116             int aId = aCommandsMap[aCommand];
1117             aToolbars[aName].append(aId);
1118             aKnownCommands.append(aId);
1119           }
1120         }
1121       }
1122     }
1123   }
1124   // Find new and obsolete commands
1125   QIntList aNewCommands = myActionsList;
1126   foreach(int aId, myActionsList) {
1127     if (aKnownCommands.contains(aId)) {
1128       aKnownCommands.removeAll(aId);
1129       aNewCommands.removeAll(aId);
1130     }
1131   }
1132   if (aNewCommands.size() > 0) {
1133     // Add new commands to toolbars structure
1134     QStringList aKeys = myToolbars.keys();
1135     foreach(int aNewId, aNewCommands) {
1136       foreach(QString aName, aKeys) {
1137         if (myToolbars[aName].contains(aNewId)) {
1138           if (!aToolbars.contains(aName)) {
1139             aToolbars[aName] = QIntList();
1140           }
1141           aToolbars[aName].append(aNewId);
1142         }
1143       }
1144     }
1145   }
1146   if (aKnownCommands.size() > 0) {
1147     // Remove obsolete commands from the toolbars structure
1148     QStringList aKeys = aToolbars.keys();
1149     foreach(int aOldId, aKnownCommands) {
1150       foreach(QString aName, aKeys) {
1151         if (aToolbars[aName].contains(aOldId)) {
1152           aToolbars[aName].removeAll(aOldId);
1153           if (aToolbars[aName].size() == 0)
1154             aToolbars.remove(aName);
1155         }
1156       }
1157     }
1158   }
1159   updateToolbars(aToolbars);
1160   myIsToolbarsModified = false;
1161 }
1162
1163
1164 QIntList SHAPERGUI::getFreeCommands() const
1165 {
1166   QIntList aFreeCommands;
1167   QtxActionToolMgr* aMgr = toolMgr();
1168   QAction* anAction;
1169   int aId;
1170   QMap<QString, QIntList>::const_iterator aIt;
1171   QIntList aShaperActions = shaperActions();
1172   foreach(int aCmd, aShaperActions) {
1173     anAction = action(aCmd);
1174     aId = aMgr->actionId(anAction);
1175     if (!aMgr->containsAction(aId))
1176       aFreeCommands.append(aCmd);
1177   }
1178   return aFreeCommands;
1179 }
1180
1181 void SHAPERGUI::resetToolbars()
1182 {
1183   if (!myDefaultToolbars.isEmpty())
1184     updateToolbars(myDefaultToolbars);
1185   myIsToolbarsModified = false;
1186   SUIT_ResourceMgr* aResMgr = application()->resourceMgr();
1187   aResMgr->remove(ToolbarsSection);
1188 }
1189
1190 void SHAPERGUI::publishToStudy()
1191 {
1192   if (isActiveModule())
1193     myWorkshop->module()->launchOperation("PublishToStudy", false);
1194 }