]> SALOME platform Git repositories - modules/shaper.git/blob - src/ModuleBase/ModuleBase_ModelWidget.cpp
Salome HOME
721c6c7b0470afd3ce84db7a0ee2517d58a6f7d2
[modules/shaper.git] / src / ModuleBase / ModuleBase_ModelWidget.cpp
1 // Copyright (C) 2014-20xx CEA/DEN, EDF R&D
2
3 // File:        ModuleBase_ModelWidget.cpp
4 // Created:     25 Apr 2014
5 // Author:      Natalia ERMOLAEVA
6
7 #include "ModuleBase_ModelWidget.h"
8
9 #include <ModelAPI_Data.h>
10 #include <ModelAPI_Attribute.h>
11 #include <ModelAPI_Events.h>
12 #include <ModelAPI_Session.h>
13
14 #include <Config_Keywords.h>
15 #include <Config_WidgetAPI.h>
16
17 #include <Events_Loop.h>
18
19 #include <QEvent>
20 #include <QGraphicsDropShadowEffect>
21 #include <QColor>
22 #include <QLabel>
23 #include <QFocusEvent>
24
25 ModuleBase_ModelWidget::ModuleBase_ModelWidget(QWidget* theParent,
26                                                const Config_WidgetAPI* theData,
27                                                const std::string& theParentId)
28     : QWidget(theParent),
29       myParentId(theParentId),
30       myIsEditing(false),
31       myState(Stored)
32 {
33   myDefaultValue = theData->getProperty(ATTR_DEFAULT);
34   myUseReset = theData->getBooleanAttribute(ATTR_USE_RESET, true);
35   myIsComputedDefault = theData->getProperty(ATTR_DEFAULT) == DOUBLE_WDG_DEFAULT_COMPUTED;
36   myAttributeID = theData ? theData->widgetId() : "";
37   myIsObligatory = theData->getBooleanAttribute(ATTR_OBLIGATORY, true);
38
39   connect(this, SIGNAL(valuesChanged()), this, SLOT(onWidgetValuesChanged()));
40   connect(this, SIGNAL(valuesModified()), this, SLOT(onWidgetValuesModified()));
41 }
42
43 bool ModuleBase_ModelWidget::reset()
44 {
45   bool aResult = resetCustom();
46   if (aResult)
47     setValueState(Reset);
48
49   return aResult;
50 }
51
52 bool ModuleBase_ModelWidget::isInitialized(ObjectPtr theObject) const
53 {
54   return theObject->data()->attribute(attributeID())->isInitialized();
55 }
56
57 void ModuleBase_ModelWidget::enableFocusProcessing()
58 {
59   QList<QWidget*> aMyControls = getControls();
60   foreach(QWidget*  eachControl, aMyControls) {
61     if (myIsObligatory) {
62       eachControl->setFocusPolicy(Qt::StrongFocus);
63       eachControl->installEventFilter(this);
64     }
65     else {
66       eachControl->setFocusPolicy(Qt::NoFocus);
67     }
68   }
69 }
70
71 void ModuleBase_ModelWidget::setHighlighted(bool isHighlighted)
72 {
73   QList<QWidget*> aWidgetList = getControls();
74   foreach(QWidget* aWidget, aWidgetList) {
75     QLabel* aLabel = qobject_cast<QLabel*>(aWidget);
76     // We won't set the effect to QLabels - it looks ugly
77     if(aLabel) continue;
78     if(isHighlighted) {
79       // If effect is the installed on a different widget, setGraphicsEffect() will
80       // remove the effect from the widget and install it on this widget.
81       // That's why we create a new effect for each widget
82       QGraphicsDropShadowEffect* aGlowEffect = new QGraphicsDropShadowEffect();
83       aGlowEffect->setOffset(.0);
84       aGlowEffect->setBlurRadius(10.0);
85       aGlowEffect->setColor(QColor(0, 170, 255)); // Light-blue color, #00AAFF
86       aWidget->setGraphicsEffect(aGlowEffect);
87     } else {
88       QGraphicsEffect* anEffect = aWidget->graphicsEffect();
89       if(anEffect)
90         anEffect->deleteLater();
91       aWidget->setGraphicsEffect(NULL);
92     }
93   }
94 }
95
96 void ModuleBase_ModelWidget::setFeature(const FeaturePtr& theFeature, const bool theToStoreValue)
97 {
98   myFeature = theFeature;
99   if (theToStoreValue)
100     storeValue();
101 }
102
103 bool ModuleBase_ModelWidget::focusTo()
104 {
105   QList<QWidget*> aControls = getControls();
106   QList<QWidget*>::const_iterator anIt = aControls.begin(), aLast = aControls.end();
107   bool isFocusAccepted = false;
108   for (; anIt != aLast && !isFocusAccepted; anIt++) {
109     QWidget* aWidget = *anIt;
110     if (aWidget && aWidget->focusPolicy() != Qt::NoFocus) {
111       aWidget->setFocus();
112       isFocusAccepted = true;
113     }
114   }
115   return isFocusAccepted;
116 }
117
118 void ModuleBase_ModelWidget::activate()
119 {
120   // the control value is stored to the mode by the focus in on the widget
121   // we need the value is initialized in order to enable the apply button in the property panel.
122   // It should happens in the creation mode only because all fields are filled in the edition mode
123   if (!isEditingMode()) {
124     AttributePtr anAttribute = myFeature->data()->attribute(myAttributeID);
125     if (anAttribute.get() != NULL && !anAttribute->isInitialized()) {
126       if (isComputedDefault()) {
127         if (myFeature->compute(myAttributeID)) {
128           restoreValue();
129         }
130       }
131       else {
132         storeValue();
133       }
134     }
135   }
136   activateCustom();
137 }
138
139 QWidget* ModuleBase_ModelWidget::getControlAcceptingFocus(const bool isFirst)
140 {
141   QWidget* aControl = 0;
142
143   QList<QWidget*> aControls = getControls();
144   int aSize = aControls.size();
145
146   if (isFirst) {
147     for (int i = 0; i < aSize && !aControl; i++)  {
148       if (aControls[i]->focusPolicy() != Qt::NoFocus)
149         aControl = aControls[i];
150     }
151   }
152   else {
153     for (int i = aSize - 1; i >= 0 && !aControl; i--)  {
154       if (aControls[i]->focusPolicy() != Qt::NoFocus)
155         aControl = aControls[i];
156     }
157   }
158   return aControl;
159 }
160
161 void ModuleBase_ModelWidget::setDefaultValue(const std::string& theValue)
162 {
163   myDefaultValue = theValue;
164 }
165
166 bool ModuleBase_ModelWidget::storeValue()
167 {
168   setValueState(Stored);
169
170   emit beforeValuesChanged();
171   bool isDone = storeValueCustom();
172   emit afterValuesChanged();
173
174   return isDone;
175 }
176
177 void ModuleBase_ModelWidget::setValueState(const ValueState& theState)
178 {
179   if (myState == theState)
180     return;
181
182   myState = theState;
183   emit valueStateChanged();
184 }
185
186 bool ModuleBase_ModelWidget::restoreValue()
187 {
188   emit beforeValuesRestored();
189   bool isDone = restoreValueCustom();
190   emit afterValuesRestored();
191
192   return isDone;
193 }
194
195 void ModuleBase_ModelWidget::storeValueByApply()
196 {
197   // do not emit signal about update the currenty feature object
198   // in order to do not perform additional redisplay in the viewer.
199   // It should happens by finish operation of the apply action
200   storeValueCustom();
201 }
202
203 void ModuleBase_ModelWidget::updateObject(ObjectPtr theObj)
204 {
205   blockUpdateViewer(true);
206
207   Events_Loop::loop()->flush(Events_Loop::eventByName(EVENT_OBJECT_UPDATED));
208
209   blockUpdateViewer(false);
210 }
211
212 void ModuleBase_ModelWidget::moveObject(ObjectPtr theObj)
213 {
214   //blockUpdateViewer(true);
215
216   static Events_ID anEvent = Events_Loop::eventByName(EVENT_OBJECT_MOVED);
217   ModelAPI_EventCreator::get()->sendUpdated(theObj, anEvent);
218   Events_Loop::loop()->flush(anEvent);
219
220   //blockUpdateViewer(false);
221 }
222
223 bool ModuleBase_ModelWidget::processEnter()
224 {
225   return false;
226 }
227
228 bool ModuleBase_ModelWidget::eventFilter(QObject* theObject, QEvent *theEvent)
229 {
230   QWidget* aWidget = qobject_cast<QWidget*>(theObject);
231   if (theEvent->type() == QEvent::FocusIn) {
232     #ifdef _DEBUG
233     // The following two lines are for debugging purpose only
234     QFocusEvent* aFocusEvent = dynamic_cast<QFocusEvent*>(theEvent);
235     bool isWinFocus = aFocusEvent->reason() == Qt::ActiveWindowFocusReason;
236     #endif
237     if (getControls().contains(aWidget)) {
238       emit focusInWidget(this);
239     }
240   }
241   // pass the event on to the parent class
242
243   return QObject::eventFilter(theObject, theEvent);
244 }
245
246 //**************************************************************
247 void ModuleBase_ModelWidget::onWidgetValuesChanged()
248 {
249   storeValue();
250 }
251
252 //**************************************************************
253 void ModuleBase_ModelWidget::onWidgetValuesModified()
254 {
255   setValueState(Modified);
256 }
257
258 //**************************************************************
259 void ModuleBase_ModelWidget::blockUpdateViewer(const bool theValue)
260 {
261   // the viewer update should be blocked in order to avoid the temporary feature content
262   // when the solver processes the feature, the redisplay message can be flushed
263   // what caused the display in the viewer preliminary states of object
264   // e.g. fillet feature, angle value change
265   std::shared_ptr<Events_Message> aMsg;
266   if (theValue) {
267     aMsg = std::shared_ptr<Events_Message>(
268         new Events_Message(Events_Loop::eventByName(EVENT_UPDATE_VIEWER_BLOCKED)));
269   }
270   else {
271     // the viewer update should be unblocked
272     aMsg = std::shared_ptr<Events_Message>(
273         new Events_Message(Events_Loop::eventByName(EVENT_UPDATE_VIEWER_UNBLOCKED)));
274   }
275   Events_Loop::loop()->send(aMsg);
276 }