Salome HOME
#1404 Random crash with Shaper: AIS presentations: operation prs, result sketch prs...
[modules/shaper.git] / src / ParametersPlugin / ParametersPlugin_WidgetParamsMgr.cpp
1 // Copyright (C) 2014-20xx CEA/DEN, EDF R&D -->
2
3 // File:        ParametersPlugin_WidgetParamsMgr.cpp
4 // Created:     11 Apr 2016
5 // Author:      Vitaly SMETANNIKOV
6
7 #include "ParametersPlugin_WidgetParamsMgr.h"
8 #include "ParametersPlugin_Parameter.h"
9 #include "ParametersPlugin_Validators.h"
10
11 #include <ModelAPI_ResultParameter.h>
12 #include <ModelAPI_AttributeString.h>
13 #include <ModelAPI_AttributeRefList.h>
14 #include <ModelAPI_AttributeDouble.h>
15 #include <ModelAPI_AttributeInteger.h>
16 #include <ModelAPI_Events.h>
17 #include <ModelAPI_Session.h>
18
19 #include <ModuleBase_Tools.h>
20
21 #include <Events_Loop.h>
22
23 #include <QLayout>
24 #include <QTreeWidget>
25 #include <QPushButton>
26 #include <QToolButton>
27 #include <QStyledItemDelegate>
28 #include <QPainter>
29 #include <QMessageBox>
30 #include <QTimer>
31
32 enum ColumnType {
33   Col_Name,
34   Col_Equation,
35   Col_Result,
36   Col_Comment
37 };
38
39 const char* NoName = "<NoName>";
40 const char* NoValue = "<NoValue>";
41
42 class ParametersPlugin_ItemDelegate : public QStyledItemDelegate
43 {
44 public:
45   ParametersPlugin_ItemDelegate(QObject* thaParent) : 
46       QStyledItemDelegate(thaParent) {}
47
48   virtual void paint(QPainter* painter, 
49     const QStyleOptionViewItem& option, 
50     const QModelIndex& index ) const;
51   
52   //virtual QWidget* createEditor(QWidget* parent, 
53   //                              const QStyleOptionViewItem& option, 
54   //                              const QModelIndex& index) const;
55
56   bool isEditable(const QModelIndex& theIndex) const;
57 };
58
59 bool ParametersPlugin_ItemDelegate::isEditable(const QModelIndex& theIndex) const
60 {
61   QModelIndex aParent = theIndex.parent();
62   if (aParent.isValid() && (aParent.row() == 0)) {
63     if (theIndex.column() == 2)
64       return false;
65   } else
66     return false;
67   return true;
68 }
69
70 void ParametersPlugin_ItemDelegate::paint(QPainter* painter, 
71                                           const QStyleOptionViewItem& option, 
72                                           const QModelIndex& index ) const
73 {
74   QBrush aBrush = painter->brush();
75   QPen aPen = painter->pen();
76   if (!isEditable(index))
77     painter->setBrush(Qt::lightGray);
78
79   painter->setPen(Qt::darkGray);
80   painter->drawRect(option.rect);
81   painter->setPen(aPen);
82   
83   //QString aText = index.data().toString();
84   //if ((aText == NoName) || (aText == NoValue))
85   //  painter->setPen(Qt::red);
86
87   QStyledItemDelegate::paint(painter, option, index);
88
89   //painter->setPen(aPen);
90   painter->setBrush(aBrush);
91 }
92
93
94 //QWidget* ParametersPlugin_ItemDelegate::createEditor(QWidget* parent, 
95 //                                                    const QStyleOptionViewItem& option, 
96 //                                                    const QModelIndex& index) const
97 //{
98 //  QWidget* aWgt = QStyledItemDelegate::createEditor(parent, option, index); 
99 //  aWgt->setMinimumSize(option.rect.width() - option.decorationSize.width(), 
100 //                       option.rect.height());
101 //  return aWgt;
102 //}
103
104
105 /////////////////////////////////////////////////////////////////////////////////////////////////
106
107 ParametersPlugin_WidgetParamsMgr::ParametersPlugin_WidgetParamsMgr(QWidget* theParent, const Config_WidgetAPI* theData)
108   : ModuleBase_ModelWidget(theParent, theData)
109 {
110   QVBoxLayout* aLayout = new QVBoxLayout(this);
111
112   myTable = new QTreeWidget(this);
113   myTable->setColumnCount(4);
114   QStringList aHeaders;
115   aHeaders << tr("Name") << tr("Equation") << tr("Result") << tr("Comment");
116   myTable->setHeaderLabels(aHeaders);
117   myTable->setColumnWidth(Col_Name, 200);
118   myTable->setColumnWidth(Col_Equation, 100);
119   myTable->setColumnWidth(Col_Result, 80);
120   myTable->setColumnWidth(Col_Comment, 200);
121   myTable->setMinimumWidth(600);
122   myTable->setEditTriggers(QAbstractItemView::NoEditTriggers);
123   myTable->setSelectionMode(QAbstractItemView::SingleSelection);
124
125   connect(myTable, SIGNAL(doubleClicked(const QModelIndex&)),
126           SLOT(onDoubleClick(const QModelIndex&)));
127   connect(myTable, SIGNAL(itemSelectionChanged()), SLOT(onSelectionChanged()));
128
129   myDelegate = new ParametersPlugin_ItemDelegate(myTable);
130   connect(myDelegate, SIGNAL(closeEditor(QWidget*, QAbstractItemDelegate::EndEditHint)), 
131           SLOT(onCloseEditor(QWidget*, QAbstractItemDelegate::EndEditHint)));
132
133   myTable->setItemDelegate(myDelegate);
134   aLayout->addWidget(myTable);
135
136   // Define root nodes
137   QStringList aNames;
138   aNames<<tr("Parameters");
139   myParameters = new QTreeWidgetItem(aNames);
140   myTable->addTopLevelItem(myParameters);
141
142   aNames.clear();
143   aNames<<tr("Features");
144   myFeatures = new QTreeWidgetItem(aNames);
145   myTable->addTopLevelItem(myFeatures);
146
147   QHBoxLayout* aBtnLayout = new QHBoxLayout(this);
148
149   myUpBtn = new QToolButton(this);
150   myUpBtn->setArrowType(Qt::UpArrow);
151   connect(myUpBtn, SIGNAL(clicked(bool)), SLOT(onUp()));
152   aBtnLayout->addWidget(myUpBtn);
153
154   myDownBtn = new QToolButton(this);
155   myDownBtn->setArrowType(Qt::DownArrow);
156   connect(myDownBtn, SIGNAL(clicked(bool)), SLOT(onDown()));
157   aBtnLayout->addWidget(myDownBtn);
158
159   aBtnLayout->addStretch();
160
161   myAddBtn = new QPushButton(tr("Add"), this);
162   connect(myAddBtn, SIGNAL(clicked(bool)), SLOT(onAdd()));
163   aBtnLayout->addWidget(myAddBtn);
164
165   myInsertBtn = new QPushButton(tr("Insert"), this);
166   connect(myInsertBtn, SIGNAL(clicked(bool)), SLOT(onInsert()));
167   aBtnLayout->addWidget(myInsertBtn);
168
169   myRemoveBtn = new QPushButton(tr("Remove"), this);
170   connect(myRemoveBtn, SIGNAL(clicked(bool)), SLOT(onRemove()));
171   aBtnLayout->addWidget(myRemoveBtn);
172
173   aLayout->addLayout(aBtnLayout);
174
175   onSelectionChanged();
176 }
177
178 QList<QWidget*> ParametersPlugin_WidgetParamsMgr::getControls() const
179 {
180   QList<QWidget*> aList;
181
182   return aList;
183 }
184
185 void ParametersPlugin_WidgetParamsMgr::selectItemScroll(QTreeWidgetItem* aItem)
186 {
187   myTable->clearSelection();
188   QModelIndex aParent = myTable->model()->index(0, 0);
189   int aChildIdx = myParameters->indexOfChild(aItem);
190   QModelIndex aIndex = myTable->model()->index(aChildIdx, Col_Name, aParent);
191   myTable->selectionModel()->select(aIndex, 
192     QItemSelectionModel::SelectCurrent | QItemSelectionModel::Rows);
193   myTable->scrollToItem(aItem);
194 }
195
196
197 bool ParametersPlugin_WidgetParamsMgr::storeValueCustom()
198 {
199   ParametersPlugin_ExpressionValidator aValidator;
200   std::list<std::string> aArgs;
201   std::string aAttrId = ParametersPlugin_Parameter::VARIABLE_ID();
202   std::string aErr;
203   int aId = 0;
204   foreach(FeaturePtr aFeature, myParametersList) {
205     if (!aValidator.isValid(aFeature->attribute(aAttrId), aArgs, aErr)) {
206       QMessageBox::warning(this, tr("Warning"), aErr.c_str());
207       selectItemScroll(myParameters->child(aId));
208       return false;
209     }
210     aId++;
211   }
212   return true;
213 }
214
215 bool ParametersPlugin_WidgetParamsMgr::restoreValueCustom()
216 {
217   return true;
218 }
219
220 void ParametersPlugin_WidgetParamsMgr::activateCustom()
221 {
222   FeaturePtr aFeature = feature();
223   DocumentPtr aDoc = aFeature->document();
224   int aNbParam = aDoc->size(ModelAPI_ResultParameter::group());
225   ObjectPtr aObj;
226   QTreeWidgetItem* aItem;
227   ResultParameterPtr aParam;
228   FeaturePtr aParamFeature;
229   for (int i = 0; i < aNbParam; i++) {
230     aObj = aDoc->object(ModelAPI_ResultParameter::group(), i);
231     aParam = std::dynamic_pointer_cast<ModelAPI_ResultParameter>(aObj);
232     if (aParam.get()) {
233       // Set parameter feature
234       aParamFeature = ModelAPI_Feature::feature(aParam);
235
236       QStringList aValues;
237       aValues << aParamFeature->string(ParametersPlugin_Parameter::VARIABLE_ID())->value().c_str();
238       aValues << aParamFeature->string(ParametersPlugin_Parameter::EXPRESSION_ID())->value().c_str();
239
240       AttributeDoublePtr aValueAttribute = aParam->data()->real(ModelAPI_ResultParameter::VALUE());
241       aValues << QString::number(aValueAttribute->value());
242
243       aValues << aParamFeature->string(ParametersPlugin_Parameter::COMMENT_ID())->value().c_str();
244
245       aItem = new QTreeWidgetItem(aValues);
246       aItem->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEditable | Qt::ItemIsEnabled);
247       myParameters->addChild(aItem);
248
249       myParametersList.append(aParamFeature);
250
251       // Set features where the parameter is used
252       const std::set<std::shared_ptr<ModelAPI_Attribute>>& aRefs = aParam->data()->refsToMe();
253       std::set<std::shared_ptr<ModelAPI_Attribute> >::const_iterator aIt;
254       for(aIt = aRefs.cbegin(); aIt != aRefs.cend(); aIt++) {
255         std::shared_ptr<ModelAPI_Attribute> aAttr = (*aIt);
256         FeaturePtr aReferenced = std::dynamic_pointer_cast<ModelAPI_Feature>(aAttr->owner());
257         if (aReferenced.get()) {
258           QStringList aValNames;
259           aValNames << aReferenced->data()->name().c_str();
260
261           AttributeDoublePtr aDouble = std::dynamic_pointer_cast<ModelAPI_AttributeDouble>(aAttr);
262           if (aDouble.get()) {
263             aValNames << aDouble->text().c_str();
264             aValNames << QString::number(aDouble->value());
265           } else {
266             AttributeIntegerPtr aInt = std::dynamic_pointer_cast<ModelAPI_AttributeInteger>(aAttr);
267             if (aInt.get()) {
268               aValNames << aInt->text().c_str();
269               aValNames << QString::number(aInt->value());
270             }
271           }
272
273           aItem = new QTreeWidgetItem(aValNames);
274           myFeatures->addChild(aItem);
275         }
276       }
277     }
278   }
279   myFeatures->setExpanded(true);
280   myParameters->setExpanded(true);
281 }
282
283
284 void ParametersPlugin_WidgetParamsMgr::onDoubleClick(const QModelIndex& theIndex)
285 {
286   if (myDelegate->isEditable(theIndex)) {
287     myTable->edit(theIndex);
288     myEditingIndex = theIndex;
289   }
290 }
291
292 void ParametersPlugin_WidgetParamsMgr::onCloseEditor(QWidget* theEditor, 
293                                                      QAbstractItemDelegate::EndEditHint theHint)
294 {
295   FeaturePtr aFeature = myParametersList.at(myEditingIndex.row());
296   QTreeWidgetItem* aItem = myParameters->child(myEditingIndex.row());
297   int aColumn = myEditingIndex.column();
298   QString aText = aItem->text(aColumn);
299   bool isModified = false;
300
301   switch (aColumn) {
302   case Col_Name:
303     {
304       AttributeStringPtr aStringAttr = aFeature->string(ParametersPlugin_Parameter::VARIABLE_ID());
305       if (!aText.isEmpty()) {
306         if (hasName(aText)) {
307           myMessage = tr("Name %1 already exists.").arg(aText);
308           aItem->setText(Col_Name, aStringAttr->value().c_str());
309           QTimer::singleShot(50, this, SLOT(sendWarning()));
310           return;
311         }
312         aStringAttr->setValue(aText.toStdString());
313         isModified = true;
314       } else {
315         aItem->setText(Col_Name, aStringAttr->value().c_str());
316       }
317     }
318     break;
319   case Col_Equation:
320     {
321       AttributeStringPtr aStringAttr = aFeature->string(ParametersPlugin_Parameter::EXPRESSION_ID());
322       if (!aText.isEmpty()) {
323         if (aText != aStringAttr->value().c_str()) {
324           aStringAttr->setValue(aText.toStdString());
325           isModified = true;
326         }
327       } else {
328         aItem->setText(Col_Equation, aStringAttr->value().c_str());
329       }
330     }
331     break;
332   case Col_Comment:
333     {
334       AttributeStringPtr aStringAttr = aFeature->string(ParametersPlugin_Parameter::COMMENT_ID());
335       aStringAttr->setValue(aText.toStdString());
336       isModified = true;
337     }
338     break;
339   }
340
341   if (!isModified)
342     return;
343   Events_Loop* aLoop = Events_Loop::loop();
344   aLoop->flush(Events_Loop::eventByName(EVENT_OBJECT_UPDATED));
345
346   ResultParameterPtr aResult = 
347     std::dynamic_pointer_cast<ModelAPI_ResultParameter>(aFeature->firstResult());
348   if (aResult.get()) {
349     AttributeDoublePtr aValueAttribute = 
350       aResult->data()->real(ModelAPI_ResultParameter::VALUE());
351     aItem->setText(Col_Result, QString::number(aValueAttribute->value()));
352   }
353   myEditingIndex = QModelIndex();
354   updateFeaturesPart();
355 }
356
357 void ParametersPlugin_WidgetParamsMgr::updateFeaturesPart()
358 {
359   std::shared_ptr<ModelAPI_ResultParameter> aParam;
360   int i = 0;
361   foreach(FeaturePtr aFeature, myParametersList) {
362     aParam = std::dynamic_pointer_cast<ModelAPI_ResultParameter>(aFeature->firstResult());
363     const std::set<std::shared_ptr<ModelAPI_Attribute>>& aRefs = aParam->data()->refsToMe();
364     std::set<std::shared_ptr<ModelAPI_Attribute> >::const_iterator aIt;
365     for(aIt = aRefs.cbegin(); aIt != aRefs.cend(); aIt++) {
366       std::shared_ptr<ModelAPI_Attribute> aAttr = (*aIt);
367       FeaturePtr aReferenced = std::dynamic_pointer_cast<ModelAPI_Feature>(aAttr->owner());
368       if (aReferenced.get()) {
369         QStringList aValNames;
370         aValNames << aReferenced->data()->name().c_str();
371
372         AttributeDoublePtr aDouble = std::dynamic_pointer_cast<ModelAPI_AttributeDouble>(aAttr);
373         if (aDouble.get()) {
374           aValNames << aDouble->text().c_str();
375           aValNames << QString::number(aDouble->value());
376         } else {
377           AttributeIntegerPtr aInt = std::dynamic_pointer_cast<ModelAPI_AttributeInteger>(aAttr);
378           if (aInt.get()) {
379             aValNames << aInt->text().c_str();
380             aValNames << QString::number(aInt->value());
381           }
382         }
383
384         QTreeWidgetItem* aItem = myFeatures->child(i++);
385         for(int i = 0; i < aValNames.count(); i++)
386           aItem->setText(i, aValNames.at(i));
387       }
388     }
389   }
390 }
391
392 FeaturePtr ParametersPlugin_WidgetParamsMgr::createParameter() const
393 {
394   SessionPtr aMgr = ModelAPI_Session::get();
395   std::shared_ptr<ModelAPI_Document> aDoc = aMgr->activeDocument();
396
397   FeaturePtr aFeature = aDoc->addFeature(ParametersPlugin_Parameter::ID());
398   if (aFeature.get()) {
399     Events_Loop::loop()->flush(Events_Loop::eventByName(EVENT_OBJECT_CREATED));
400     Events_Loop::loop()->flush(Events_Loop::eventByName(EVENT_OBJECT_UPDATED));
401   }
402   return aFeature;
403 }
404
405
406 QTreeWidgetItem* ParametersPlugin_WidgetParamsMgr::createNewItem() const
407 {
408   QStringList aValues;
409   aValues << NoName;
410   aValues << NoValue;
411
412   QTreeWidgetItem* aItem = new QTreeWidgetItem(aValues);
413   aItem->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEditable | Qt::ItemIsEnabled);
414   return aItem;
415 }
416
417
418 void ParametersPlugin_WidgetParamsMgr::onAdd()
419 {
420   FeaturePtr aFeature = createParameter();
421   if (!aFeature.get())
422     return;
423
424   QTreeWidgetItem* aItem = createNewItem();
425   myParameters->addChild(aItem);
426   myParametersList.append(aFeature);
427       
428   myTable->scrollToItem(aItem);
429 }
430
431 QTreeWidgetItem* ParametersPlugin_WidgetParamsMgr::selectedItem() const
432 {
433   QList<QTreeWidgetItem*> aItemsList = myTable->selectedItems();
434   if (aItemsList.count() == 0)
435     return 0;
436
437   QTreeWidgetItem* aCurrentItem = aItemsList.first();
438   if (aCurrentItem->parent() != myParameters)
439     return 0;
440
441   return aCurrentItem;
442 }
443
444
445 void ParametersPlugin_WidgetParamsMgr::onInsert()
446 {
447   QTreeWidgetItem* aCurrentItem = selectedItem();
448   if (!aCurrentItem)
449     return;
450
451   SessionPtr aMgr = ModelAPI_Session::get();
452   std::shared_ptr<ModelAPI_Document> aDoc = aMgr->activeDocument();
453
454   FeaturePtr aNewFeature = createParameter();
455   if (!aNewFeature.get())
456     return;
457
458   QTreeWidgetItem* aItem = createNewItem();
459   int aCurrentPos = myParameters->indexOfChild(aCurrentItem);
460   if (aCurrentPos == 0) {
461     aDoc->moveFeature(aNewFeature, FeaturePtr());
462   } else {
463     FeaturePtr aCurFeature = myParametersList.at(aCurrentPos - 1);
464     aDoc->moveFeature(aNewFeature, aCurFeature);
465   }
466   myParametersList.insert(aCurrentPos, aNewFeature);
467   myParameters->insertChild(aCurrentPos, aItem);
468 }
469
470 void ParametersPlugin_WidgetParamsMgr::onRemove()
471 {
472   QTreeWidgetItem* aCurrentItem = selectedItem();
473   if (!aCurrentItem)
474     return;
475
476   SessionPtr aMgr = ModelAPI_Session::get();
477   std::shared_ptr<ModelAPI_Document> aDoc = aMgr->activeDocument();
478
479   int aCurrentPos = myParameters->indexOfChild(aCurrentItem);
480   FeaturePtr aCurFeature = myParametersList.at(aCurrentPos);
481
482   QObjectPtrList anObjects;
483   anObjects.append(aCurFeature);
484
485   std::set<FeaturePtr> aDirectRefFeatures, aIndirectRefFeatures;
486   ModuleBase_Tools::findReferences(anObjects, aDirectRefFeatures, aIndirectRefFeatures);
487
488   bool doDeleteReferences = true;
489   if (ModuleBase_Tools::isDeleteFeatureWithReferences(anObjects, aDirectRefFeatures, 
490       aIndirectRefFeatures, this, doDeleteReferences)) {
491     myParametersList.removeOne(aCurFeature);
492     myParameters->removeChild(aCurrentItem);
493     aDoc->removeFeature(aCurFeature);
494   }
495 }
496
497 void ParametersPlugin_WidgetParamsMgr::onUp()
498 {
499   QTreeWidgetItem* aCurrentItem = selectedItem();
500   if (!aCurrentItem)
501     return;
502
503   QString aName = aCurrentItem->text(0);
504
505   int aCurrentPos = myParameters->indexOfChild(aCurrentItem);
506   if (aCurrentPos == 0)
507     return;
508   FeaturePtr aCurFeature = myParametersList.at(aCurrentPos);
509
510   std::string aNm = aCurFeature->data()->name();
511
512   SessionPtr aMgr = ModelAPI_Session::get();
513   std::shared_ptr<ModelAPI_Document> aDoc = aMgr->activeDocument();
514
515   FeaturePtr aa = myParametersList.at(aCurrentPos - 1);
516   std::string aN = aa->data()->name();
517
518   if (aCurrentPos == 1)
519     aDoc->moveFeature(aCurFeature, FeaturePtr());
520   else
521     aDoc->moveFeature(aCurFeature, myParametersList.at(aCurrentPos - 2));
522
523   myParametersList.removeOne(aCurFeature);
524   myParametersList.insert(aCurrentPos - 1, aCurFeature);
525
526   myParameters->removeChild(aCurrentItem);
527   myParameters->insertChild(aCurrentPos - 1, aCurrentItem);
528
529   selectItemScroll(aCurrentItem);
530 }
531
532 void ParametersPlugin_WidgetParamsMgr::onDown()
533 {
534   QTreeWidgetItem* aCurrentItem = selectedItem();
535   if (!aCurrentItem)
536     return;
537
538   int aCurrentPos = myParameters->indexOfChild(aCurrentItem);
539   if (aCurrentPos == (myParametersList.count() - 1))
540     return;
541   FeaturePtr aCurFeature = myParametersList.at(aCurrentPos);
542
543   SessionPtr aMgr = ModelAPI_Session::get();
544   std::shared_ptr<ModelAPI_Document> aDoc = aMgr->activeDocument();
545   aDoc->moveFeature(aCurFeature, myParametersList.at(aCurrentPos + 1));
546
547   myParametersList.removeOne(aCurFeature);
548   myParametersList.insert(aCurrentPos + 1, aCurFeature);
549
550   myParameters->removeChild(aCurrentItem);
551   myParameters->insertChild(aCurrentPos + 1, aCurrentItem);
552
553   selectItemScroll(aCurrentItem);
554 }
555
556
557 bool ParametersPlugin_WidgetParamsMgr::hasName(const QString& theName) const
558 {
559   foreach(FeaturePtr aFeature, myParametersList) {
560     if (aFeature->data()->name() == theName.toStdString())
561       return true;
562   }
563   return false;
564 }
565
566 void ParametersPlugin_WidgetParamsMgr::sendWarning()
567 {
568   QMessageBox::warning(this, tr("Warning"), myMessage);
569 }
570
571 void ParametersPlugin_WidgetParamsMgr::onSelectionChanged()
572 {
573   QList<QTreeWidgetItem*> aItemsList = myTable->selectedItems();
574   bool isParameter = false;
575   foreach(QTreeWidgetItem* aItem, aItemsList) {
576     if (aItem->parent() == myParameters) {
577       isParameter = true;
578       break;
579     }
580   }
581   myInsertBtn->setEnabled(isParameter);
582   myRemoveBtn->setEnabled(isParameter);
583   myUpBtn->setEnabled(isParameter);
584   myDownBtn->setEnabled(isParameter);
585 }