Salome HOME
de74d3c4643831c3160cece278e3346aa6a589a9
[modules/shaper.git] / src / CollectionPlugin / CollectionPlugin_WidgetField.cpp
1 // Copyright (C) 2014-20xx CEA/DEN, EDF R&D
2
3 // File:        CollectionPlugin_WidgetField.cpp
4 // Created:     16 Nov 2016
5 // Author:      Vitaly SMETANNIKOV
6
7 #include "CollectionPlugin_WidgetField.h"
8 #include "CollectionPlugin_Field.h"
9
10 #include <ModuleBase_Tools.h>
11 #include <ModuleBase_IWorkshop.h>
12 #include <ModuleBase_ISelection.h>
13
14 #include <XGUI_Tools.h>
15 #include <XGUI_Workshop.h>
16 #include <XGUI_PropertyPanel.h>
17
18 #include <ModelAPI_AttributeSelectionList.h>
19 #include <ModelAPI_AttributeStringArray.h>
20 #include <ModelAPI_AttributeInteger.h>
21 #include <ModelAPI_AttributeIntArray.h>
22
23 #include <QLayout>
24 #include <QWidget>
25 #include <QFormLayout>
26 #include <QComboBox>
27 #include <QSpinBox>
28 #include <QLabel>
29 #include <QSlider>
30 #include <QTableWidget>
31 #include <QPushButton>
32 #include <QHeaderView>
33 #include <QStackedWidget>
34 #include <QValidator>
35 #include <QStyledItemDelegate>
36 #include <QLineEdit>
37 #include <QEvent>
38 #include <QMouseEvent>
39 #include <QScrollBar>
40 #include <QApplication>
41
42 const char* MYFirstCol = "Shape";
43 const char* MYTrue = "True";
44 const char* MYFalse = "False";
45
46 class DataTableItemDelegate : public QStyledItemDelegate
47 {
48 public:
49   DataTableItemDelegate(ModelAPI_AttributeTables::ValueType theType) :
50       QStyledItemDelegate() { myType = theType; }
51
52   virtual QWidget* createEditor(QWidget* theParent,
53                                 const QStyleOptionViewItem & theOption,
54                                 const QModelIndex& theIndex) const;
55
56   ModelAPI_AttributeTables::ValueType dataType() const { return myType; }
57
58   void setDataType(ModelAPI_AttributeTables::ValueType theType) { myType = theType; }
59
60 signals:
61   void startEditing();
62
63 private:
64   ModelAPI_AttributeTables::ValueType myType;
65 };
66
67 QWidget* DataTableItemDelegate::createEditor(QWidget* theParent,
68                                              const QStyleOptionViewItem & theOption,
69                                              const QModelIndex& theIndex ) const
70 {
71   if ((theIndex.column() == 0) && (theIndex.row() > 0)) {
72     QWidget* aWgt = QStyledItemDelegate::createEditor(theParent, theOption, theIndex);
73     QLineEdit* aEdt = static_cast<QLineEdit*>(aWgt);
74     aEdt->setReadOnly(true);
75     return aEdt;
76   } else {
77     QLineEdit* aLineEdt = 0;
78     switch (myType) {
79     case ModelAPI_AttributeTables::DOUBLE:
80       aLineEdt = dynamic_cast<QLineEdit*>(QStyledItemDelegate::createEditor(theParent,
81                                                                             theOption,
82                                                                             theIndex));
83       if (aLineEdt) {
84         aLineEdt->setValidator(new QDoubleValidator(aLineEdt));
85         return aLineEdt;
86       }
87       break;
88     case ModelAPI_AttributeTables::INTEGER:
89       aLineEdt = dynamic_cast<QLineEdit*>(QStyledItemDelegate::createEditor(theParent,
90                                                                             theOption,
91                                                                             theIndex));
92       if (aLineEdt) {
93         aLineEdt->setValidator(new QIntValidator(aLineEdt));
94         return aLineEdt;
95       }
96       break;
97     case ModelAPI_AttributeTables::BOOLEAN:
98       {
99         QComboBox* aBox = new QComboBox(theParent);
100         aBox->addItem(MYFalse);
101         aBox->addItem(MYTrue);
102         return aBox;
103       }
104     }
105   }
106   return QStyledItemDelegate::createEditor(theParent, theOption, theIndex);
107 }
108
109
110
111 //**********************************************************************************
112 //**********************************************************************************
113 //**********************************************************************************
114 CollectionPlugin_WidgetField::
115   CollectionPlugin_WidgetField(QWidget* theParent,
116                                ModuleBase_IWorkshop* theWorkshop,
117                                const Config_WidgetAPI* theData):
118 ModuleBase_WidgetSelector(theParent, theWorkshop, theData), myHeaderEditor(0),
119   myIsTabEdit(false), myActivation(false)
120 {
121   QVBoxLayout* aMainLayout = new QVBoxLayout(this);
122
123   // Types definition controls
124   QWidget* aTypesWgt = new QWidget(this);
125   QFormLayout* aTypesLayout = new QFormLayout(aTypesWgt);
126   aTypesLayout->setContentsMargins(0, 0, 0, 0);
127   aMainLayout->addWidget(aTypesWgt);
128
129   // Type of shapes
130   myShapeTypeCombo = new QComboBox(aTypesWgt);
131   QStringList aShapeTypes;
132   aShapeTypes << tr("Vertices") << tr("Edges") << tr("Faces")
133     << tr("Solids") << tr("Objects") << tr("Parts");
134   myShapeTypeCombo->addItems(aShapeTypes);
135   aTypesLayout->addRow(tr("Type of shapes"), myShapeTypeCombo);
136
137   // Type of field
138   myFieldTypeCombo = new QComboBox(aTypesWgt);
139   QStringList aFieldTypes;
140   aFieldTypes << tr("Boolean") << tr("Integer") << tr("Double")
141     << tr("String");
142   myFieldTypeCombo->addItems(aFieldTypes);
143   myFieldTypeCombo->setCurrentIndex(2);
144   aTypesLayout->addRow(tr("Type of field"), myFieldTypeCombo);
145
146   // Number of components
147   myNbComponentsSpn = new QSpinBox(aTypesWgt);
148   myNbComponentsSpn->setMinimum(1);
149   aTypesLayout->addRow(tr("Nb. of components"), myNbComponentsSpn);
150
151   // Steps controls
152   QFrame* aStepFrame = new QFrame(this);
153   aStepFrame->setFrameShape(QFrame::Box);
154   aStepFrame->setFrameStyle(QFrame::StyledPanel);
155   QGridLayout* aStepLayout = new QGridLayout(aStepFrame);
156   aMainLayout->addWidget(aStepFrame);
157
158   // Current step label
159   aStepLayout->addWidget(new QLabel(tr("Current step"), aStepFrame), 0, 0);
160   myCurStepLbl = new QLabel("1", aStepFrame);
161   QFont aFont = myCurStepLbl->font();
162   aFont.setBold(true);
163   myCurStepLbl->setFont(aFont);
164   aStepLayout->addWidget(myCurStepLbl, 0, 1);
165
166   // Steps slider
167   QWidget* aSliderWidget = new QWidget(aStepFrame);
168   aStepLayout->addWidget(aSliderWidget, 1, 0, 1, 2);
169   QHBoxLayout* aSliderLayout = new QHBoxLayout(aSliderWidget);
170   aSliderLayout->setContentsMargins(0, 0, 0, 0);
171
172   aSliderLayout->addWidget(new QLabel("1", aSliderWidget));
173
174   myStepSlider = new QSlider(Qt::Horizontal, aSliderWidget);
175   myStepSlider->setTickPosition(QSlider::TicksBelow);
176   myStepSlider->setRange(1, 1);
177   myStepSlider->setPageStep(myStepSlider->singleStep());
178   aSliderLayout->addWidget(myStepSlider, 1);
179
180   myMaxLbl = new QLabel("1", aSliderWidget);
181   aSliderLayout->addWidget(myMaxLbl);
182
183   // Stamp value
184   myCompNamesList << "Comp 1";
185   myStepWgt = new QStackedWidget(aStepFrame);
186   aStepLayout->addWidget(myStepWgt, 2, 0, 1, 2);
187   appendStepControls();
188
189   // Buttons below
190   QWidget* aBtnWgt = new QWidget(this);
191   aMainLayout->addWidget(aBtnWgt);
192   QHBoxLayout* aBtnLayout = new QHBoxLayout(aBtnWgt);
193   aBtnLayout->setContentsMargins(0, 0, 0, 0);
194
195   QPushButton* aAddBtn = new QPushButton(tr("Add step"), aBtnWgt);
196   aBtnLayout->addWidget(aAddBtn);
197
198   aBtnLayout->addStretch(1);
199
200   myRemoveBtn = new QPushButton(tr("Remove step"), aBtnWgt);
201   aBtnLayout->addWidget(myRemoveBtn);
202   myRemoveBtn->setEnabled(false);
203
204   connect(myNbComponentsSpn, SIGNAL(valueChanged(int)), SLOT(onNbCompChanged(int)));
205   connect(aAddBtn, SIGNAL(clicked(bool)), SLOT(onAddStep()));
206   connect(myRemoveBtn, SIGNAL(clicked(bool)), SLOT(onRemoveStep()));
207   connect(myStepSlider, SIGNAL(valueChanged(int)), SLOT(onStepMove(int)));
208   connect(myStepSlider, SIGNAL(rangeChanged(int, int)), SLOT(onRangeChanged(int, int)));
209   connect(myFieldTypeCombo, SIGNAL(currentIndexChanged(int)), SLOT(onFieldTypeChanged(int)));
210   connect(myShapeTypeCombo, SIGNAL(currentIndexChanged(int)), SLOT(onShapeTypeChanged(int)));
211   connect(qApp, SIGNAL(focusChanged(QWidget*, QWidget*)), SLOT(onFocusChanged(QWidget*, QWidget*)));
212 }
213
214 //**********************************************************************************
215 void CollectionPlugin_WidgetField::appendStepControls()
216 {
217   QWidget* aWidget = new QWidget(myStepWgt);
218   QGridLayout* aStepLayout = new QGridLayout(aWidget);
219   aStepLayout->setContentsMargins(0, 0, 0, 0);
220
221   aStepLayout->addWidget(new QLabel(tr("Stamp"), aWidget), 0, 0);
222
223   QSpinBox* aStampSpn = new QSpinBox(aWidget);
224   aStepLayout->addWidget(aStampSpn, 0, 1);
225
226   myStampSpnList.append(aStampSpn);
227
228   // Data table
229   QTableWidget* aDataTbl = new QTableWidget(1, myCompNamesList.count() + 1, aWidget);
230   aDataTbl->installEventFilter(this);
231   DataTableItemDelegate* aDelegate = 0;
232   if (myDataTblList.isEmpty())
233     aDelegate = new DataTableItemDelegate(
234       (ModelAPI_AttributeTables::ValueType) myFieldTypeCombo->currentIndex());
235   else
236     aDelegate = dynamic_cast<DataTableItemDelegate*>(myDataTblList.first()->itemDelegate());
237
238   QIntList aColWidth;
239   if (!myDataTblList.isEmpty()) {
240     QTableWidget* aFirstTable = myDataTblList.first();
241     for (int i = 0; i < aFirstTable->columnCount(); i++)
242       aColWidth.append(aFirstTable->columnWidth(i));
243   }
244   aDataTbl->setItemDelegate(aDelegate);
245   myDataTblList.append(aDataTbl);
246
247   aDataTbl->verticalHeader()->hide();
248   aDataTbl->setRowHeight(0, 25);
249   aDataTbl->setHorizontalScrollMode(QAbstractItemView::ScrollPerPixel);
250
251   connect(aDataTbl->horizontalHeader(), SIGNAL(sectionResized(int, int, int)),
252           SLOT(onColumnResize(int, int, int)));
253
254   updateHeaders(aDataTbl);
255
256   QTableWidgetItem* aItem = new QTableWidgetItem("Default value");
257   aItem->setBackgroundColor(Qt::lightGray);
258   aItem->setFlags(Qt::NoItemFlags | Qt::ItemIsEnabled);
259   aDataTbl->setItem(0, 0, aItem);
260
261   // Set default value item
262   for (int i = 0; i < myCompNamesList.count(); i++) {
263     aItem = createDefaultItem();
264     aItem->setBackgroundColor(Qt::lightGray);
265     aDataTbl->setItem(0, i + 1, aItem);
266   }
267
268   if (aColWidth.length() > 0) {
269     for (int i = 0; i < aDataTbl->columnCount(); i++)
270       aDataTbl->setColumnWidth(i, aColWidth.at(i));
271   }
272   aStepLayout->addWidget(aDataTbl, 1, 0, 1, 2);
273   connect(aDataTbl, SIGNAL(cellChanged(int, int)), SLOT(onTableEdited(int, int)));
274
275   QAbstractItemDelegate* aDel = aDataTbl->itemDelegate();
276   myStepWgt->addWidget(aWidget);
277   aDataTbl->horizontalHeader()->viewport()->installEventFilter(this);
278 }
279
280 //**********************************************************************************
281 void CollectionPlugin_WidgetField::deactivate()
282 {
283   ModuleBase_WidgetSelector::deactivate();
284   storeValueCustom();
285 }
286
287 //**********************************************************************************
288 //void CollectionPlugin_WidgetField::showEvent(QShowEvent* theEvent)
289 //{
290 //  myShapeTypeCombo->setEnabled(!isEditingMode());
291 //  myFieldTypeCombo->setEnabled(!isEditingMode());
292 //  myNbComponentsSpn->setEnabled(!isEditingMode());
293 //}
294
295 //**********************************************************************************
296 bool CollectionPlugin_WidgetField::eventFilter(QObject* theObject, QEvent* theEvent)
297 {
298   QObject* aObject = 0;
299   foreach(QTableWidget* aTable, myDataTblList) {
300     if (aTable->horizontalHeader()->viewport() == theObject) {
301       aObject = theObject;
302       break;
303     }
304   }
305   if (aObject) {
306     if (theEvent->type() == QEvent::MouseButtonDblClick) {
307       if (myHeaderEditor) { //delete previous editor
308         myHeaderEditor->deleteLater();
309         myHeaderEditor = 0;
310       }
311       QMouseEvent* aMouseEvent = static_cast<QMouseEvent*>(theEvent);
312       QHeaderView* aHeader = static_cast<QHeaderView*>(aObject->parent());
313       QTableWidget* aTable = static_cast<QTableWidget*>(aHeader->parentWidget());
314
315       int aShift = aTable->horizontalScrollBar()->value();
316       int aPos = aMouseEvent->x();
317       int aIndex = aHeader->logicalIndex(aHeader->visualIndexAt(aPos));
318       if (aIndex > 0) {
319         QRect aRect;
320         aRect.setLeft(aHeader->sectionPosition(aIndex));
321         aRect.setWidth(aHeader->sectionSize(aIndex));
322         aRect.setTop(0);
323         aRect.setHeight(aHeader->height());
324         aRect.adjust(1, 1, -1, -1);
325         aRect.translate(-aShift, 0);
326
327         myHeaderEditor = new QLineEdit(aHeader->viewport());
328         myHeaderEditor->move(aRect.topLeft());
329         myHeaderEditor->resize(aRect.size());
330         myHeaderEditor->setFrame(false);
331         QString aText = aHeader->model()->
332           headerData(aIndex, aHeader->orientation()).toString();
333         myHeaderEditor->setText(aText);
334         myHeaderEditor->setFocus();
335         myEditIndex = aIndex; //save for future use
336         myHeaderEditor->installEventFilter(this); //catch focus out event
337         //if user presses Enter it should close editor
338         connect(myHeaderEditor, SIGNAL(returnPressed()), aTable, SLOT(setFocus()));
339         myHeaderEditor->show();
340         return true;
341       }
342     }
343   } else if ((theObject == myHeaderEditor) && (theEvent->type() == QEvent::FocusOut)) {
344     QString aNewTitle = myHeaderEditor->text();
345     //save item text
346     myCompNamesList.replace(myEditIndex - 1, aNewTitle);
347     myHeaderEditor->deleteLater(); //safely delete editor
348     myHeaderEditor = 0;
349     // Store into data model
350     AttributeStringArrayPtr aStringsAttr =
351       myFeature->data()->stringArray(CollectionPlugin_Field::COMPONENTS_NAMES_ID());
352     aStringsAttr->setValue(myEditIndex - 1, aNewTitle.toStdString());
353     foreach(QTableWidget* aTable, myDataTblList) {
354       updateHeaders(aTable);
355     }
356   } else if (theEvent->type() == QEvent::FocusIn) {
357     QTableWidget* aTable = dynamic_cast<QTableWidget*>(theObject);
358     if (aTable) {
359       XGUI_Workshop* aWorkshop = XGUI_Tools::workshop(myWorkshop);
360       XGUI_PropertyPanel* aPanel = aWorkshop->propertyPanel();
361       if (aPanel->activeWidget() != this) {
362         myActivation = true;
363         aPanel->activateWidget(this, false);
364       }
365     }
366   }
367   return ModuleBase_WidgetSelector::eventFilter(theObject, theEvent);
368 }
369
370 //**********************************************************************************
371 QTableWidgetItem* CollectionPlugin_WidgetField::createDefaultItem() const
372 {
373   QTableWidgetItem* aItem = new QTableWidgetItem();
374   switch (myFieldTypeCombo->currentIndex()) {
375   case ModelAPI_AttributeTables::DOUBLE:
376   case ModelAPI_AttributeTables::INTEGER:
377     aItem->setText("0");
378     break;
379   case ModelAPI_AttributeTables::BOOLEAN:
380     aItem->setText(MYFalse);
381     break;
382   case ModelAPI_AttributeTables::STRING:
383     aItem->setText("");
384     break;
385   }
386   return aItem;
387 }
388
389 //**********************************************************************************
390 QTableWidgetItem* CollectionPlugin_WidgetField::
391   createValueItem(ModelAPI_AttributeTables::Value& theVal) const
392 {
393   QTableWidgetItem* aItem = new QTableWidgetItem();
394   aItem->setText(getValueText(theVal));
395   return aItem;
396 }
397
398 //**********************************************************************************
399 QString CollectionPlugin_WidgetField::getValueText(ModelAPI_AttributeTables::Value& theVal) const
400 {
401   switch (myFieldTypeCombo->currentIndex()) {
402   case ModelAPI_AttributeTables::DOUBLE:
403     return QString::number(theVal.myDouble);
404   case ModelAPI_AttributeTables::INTEGER:
405     return QString::number(theVal.myInt);
406   case ModelAPI_AttributeTables::BOOLEAN:
407     return theVal.myBool? MYTrue : MYFalse;
408   case ModelAPI_AttributeTables::STRING:
409     return theVal.myStr.c_str();
410   }
411   return "";
412 }
413
414
415 //**********************************************************************************
416 void CollectionPlugin_WidgetField::updateHeaders(QTableWidget* theDataTbl) const
417 {
418   QStringList aHeaders;
419   aHeaders << tr(MYFirstCol);
420   aHeaders << myCompNamesList;
421   theDataTbl->setHorizontalHeaderLabels(aHeaders);
422 }
423
424 //**********************************************************************************
425 void CollectionPlugin_WidgetField::removeStepControls()
426 {
427   int aCurWgtId = myStepWgt->currentIndex();
428   QWidget* aWgt = myStepWgt->currentWidget();
429   myStepWgt->removeWidget(aWgt);
430
431   myStampSpnList.removeAt(aCurWgtId);
432   myDataTblList.removeAt(aCurWgtId);
433   aWgt->deleteLater();
434 }
435
436 //**********************************************************************************
437 QList<QWidget*> CollectionPlugin_WidgetField::getControls() const
438 {
439   QList<QWidget*> aControls;
440   // this control will accept focus and will be highlighted in the Property Panel
441   aControls.push_back(myShapeTypeCombo);
442   aControls.push_back(myFieldTypeCombo);
443   aControls.push_back(myNbComponentsSpn);
444   return aControls;
445 }
446
447 //**********************************************************************************
448 bool CollectionPlugin_WidgetField::storeValueCustom()
449 {
450   DataPtr aData = myFeature->data();
451   // Store number of components
452   AttributeStringArrayPtr aStringsAttr =
453     aData->stringArray(CollectionPlugin_Field::COMPONENTS_NAMES_ID());
454   int aNbComps = myCompNamesList.size();
455   aStringsAttr->setSize(aNbComps);
456   for ( int i = 0; i < aNbComps; i++)
457     aStringsAttr->setValue(i, myCompNamesList.at(i).toStdString());
458
459   AttributeTablesPtr aTablesAttr = aData->tables(CollectionPlugin_Field::VALUES_ID());
460   // Store number of steps
461   int aNbSteps =  myDataTblList.size();
462
463   // Store Type of the field values
464   int aFldType = myFieldTypeCombo->currentIndex();
465
466   AttributeIntArrayPtr aStampsAttr = aData->intArray(CollectionPlugin_Field::STAMPS_ID());
467   aStampsAttr->setSize(aNbSteps);
468   // Store data
469   QTableWidget* aTable = myDataTblList.first();
470   int aRows = aTable->rowCount();
471   // first column contains selected names which should not be stored
472   int aColumns = aTable->columnCount() - 1;
473
474   aTablesAttr->setSize(aRows, aColumns, aNbSteps);
475   aTablesAttr->setType((ModelAPI_AttributeTables::ValueType)aFldType);
476   for (int i = 0; i < aNbSteps; i++) {
477     aStampsAttr->setValue(i, myStampSpnList.at(i)->value());
478     aTable = myDataTblList.at(i);
479     for (int j = 0; j < aColumns; j++) {
480       for (int k = 0; k < aRows; k++) {
481         QString aTblVal = aTable->item(k, j + 1)->text();
482         aTablesAttr->setValue(getValue(aTblVal), k, j, i);
483       }
484     }
485   }
486   updateObject(myFeature);
487   return true;
488 }
489
490 //**********************************************************************************
491 bool CollectionPlugin_WidgetField::restoreValueCustom()
492 {
493   bool isBlocked;
494   DataPtr aData = myFeature->data();
495
496   AttributeSelectionListPtr aSelList = aData->selectionList(CollectionPlugin_Field::SELECTED_ID());
497   std::string aTypeStr = aSelList->selectionType();
498   if (aTypeStr == "")
499     return false; // The attribute is not initialized
500   myShapeTypeCombo->setCurrentIndex(getSelectionType(aTypeStr));
501
502   // Get number of components
503   AttributeStringArrayPtr aStringsAttr =
504   aData->stringArray(CollectionPlugin_Field::COMPONENTS_NAMES_ID());
505
506   myCompNamesList.clear();
507   for (int i = 0; i < aStringsAttr->size(); i++) {
508     myCompNamesList.append(aStringsAttr->value(i).c_str());
509   }
510   isBlocked = myNbComponentsSpn->blockSignals(true);
511   myNbComponentsSpn->setValue(myCompNamesList.size());
512   myNbComponentsSpn->blockSignals(isBlocked);
513
514   AttributeTablesPtr aTablesAttr = aData->tables(CollectionPlugin_Field::VALUES_ID());
515   // Get number of steps
516   int aNbSteps = aTablesAttr->tables();
517   myStepSlider->setMaximum(aNbSteps);
518   //myStepSlider->setValue(1);
519   // Clear old tables
520   while (myDataTblList.count() > aNbSteps) {
521     QWidget* aWgt = myStepWgt->widget(myStepWgt->count() - 1);
522     myStepWgt->removeWidget(aWgt);
523     aWgt->deleteLater();
524
525     myStampSpnList.removeLast();
526     myDataTblList.removeLast();
527   }
528   while (myDataTblList.count() < aNbSteps)
529     appendStepControls();
530   //myStepWgt->setCurrentIndex(myStepSlider->value() - 1);
531   clearData();
532
533   // Get Type of the field values
534   isBlocked = myFieldTypeCombo->blockSignals(true);
535   myFieldTypeCombo->setCurrentIndex(aTablesAttr->type());
536   myFieldTypeCombo->blockSignals(isBlocked);
537
538   AttributeIntArrayPtr aStampsAttr = aData->intArray(CollectionPlugin_Field::STAMPS_ID());
539   // Fill data table
540   int aRows = aTablesAttr->rows();
541   int aCols = aTablesAttr->columns();
542
543   // Get width of columns
544   QIntList aColWidth;
545   QTableWidget* aFirstTable = myDataTblList.first();
546   for (int i = 0; i < aFirstTable->columnCount(); i++)
547     aColWidth.append(aFirstTable->columnWidth(i));
548
549   QTableWidgetItem* aItem = 0;
550   for (int i = 0; i < aNbSteps; i++) {
551     myStampSpnList.at(i)->setValue(aStampsAttr->value(i));
552     QTableWidget* aTable = myDataTblList.at(i);
553     isBlocked = aTable->blockSignals(true);
554     aTable->setRowCount(aRows);
555     for (int j = 0; j < aCols + 1; j++) {
556       for (int k = 0; k < aRows; k++) {
557         aItem = aTable->item(k, j);
558         if ((j == 0) && (k > 0)) {
559           // Add selection names
560           AttributeSelectionPtr aAttr = aSelList->value(k - 1);
561           if (aItem) {
562             aItem->setText(aAttr->namingName().c_str());
563           } else {
564             aItem = new QTableWidgetItem(aAttr->namingName().c_str());
565             aTable->setItem(k, j, aItem);
566           }
567         } else if (j > 0) {
568           // Add Values
569           ModelAPI_AttributeTables::Value aVal = aTablesAttr->value(k, j - 1, i);
570           if (aItem) {
571             aItem->setText(getValueText(aVal));
572           } else {
573             aItem = createValueItem(aVal);
574             if (k == 0)
575               aItem->setBackgroundColor(Qt::lightGray);
576             aTable->setItem(k, j, aItem);
577           }
578         }
579       }
580     }
581     // Restore columns width
582     for (int i = 0; i < aTable->columnCount(); i++)
583       aTable->setColumnWidth(i, aColWidth.at(i));
584
585     aTable->blockSignals(isBlocked);
586   }
587   return true;
588 }
589
590 //**********************************************************************************
591 int CollectionPlugin_WidgetField::getSelectionType(const std::string& theStr) const
592 {
593   if (theStr == "vertex")
594     return 0;
595   else if (theStr == "edge")
596     return 1;
597   else if (theStr == "face")
598     return 2;
599   else if (theStr == "solid")
600     return 3;
601   else if (theStr == "object")
602     return 4;
603   else if (theStr == "part")
604     return 5;
605   return -1;
606 }
607
608
609 //**********************************************************************************
610 std::string CollectionPlugin_WidgetField::getSelectionType(int theType) const
611 {
612   switch (theType) {
613   case 0: //"Vertices"
614     return "vertex";
615   case 1: // "Edges"
616     return "edge";
617   case 2: // "Faces"
618     return "face";
619   case 3: // "Solids"
620     return "solid";
621   case 4: // "Results"
622     return "object";
623   case 5: // "Parts"
624     return "part";
625   }
626   return "";
627 }
628
629 //**********************************************************************************
630 QIntList CollectionPlugin_WidgetField::shapeTypes() const
631 {
632   QIntList aRes;
633   switch (myShapeTypeCombo->currentIndex()) {
634   case 0: //"Vertices"
635     aRes.append(ModuleBase_Tools::shapeType("vertex"));
636     break;
637   case 1: // "Edges"
638     aRes.append(ModuleBase_Tools::shapeType("edge"));
639     break;
640   case 2: // "Faces"
641     aRes.append(ModuleBase_Tools::shapeType("face"));
642     break;
643   case 3: // "Solids"
644     aRes.append(ModuleBase_Tools::shapeType("solid"));
645     break;
646   case 4: // "Results"
647     aRes.append(ModuleBase_Tools::shapeType("object"));
648     break;
649   case 5: // "Parts"
650     // TODO: Selection mode for Parts
651     break;
652   }
653   return aRes;
654 }
655
656 //**********************************************************************************
657 ModelAPI_AttributeTables::Value CollectionPlugin_WidgetField::getValue(QString theStrVal) const
658 {
659   ModelAPI_AttributeTables::Value aVal;
660   switch (myFieldTypeCombo->currentIndex()) {
661   case ModelAPI_AttributeTables::BOOLEAN:
662     aVal.myBool = (theStrVal == MYTrue)? true : false;
663     break;
664   case ModelAPI_AttributeTables::DOUBLE:
665     aVal.myDouble = theStrVal.toDouble();
666     break;
667   case ModelAPI_AttributeTables::INTEGER:
668     aVal.myInt = theStrVal.toInt();
669     break;
670   case ModelAPI_AttributeTables::STRING:
671     aVal.myStr = theStrVal.toStdString();
672   }
673   return aVal;
674 }
675
676
677 //**********************************************************************************
678 void CollectionPlugin_WidgetField::onNbCompChanged(int theVal)
679 {
680   int aOldCol = myCompNamesList.count();
681   int aNbRows = myDataTblList.first()->rowCount();
682   int aDif = theVal - aOldCol;
683   QTableWidgetItem* aItem = 0;
684
685   while (myCompNamesList.count() != theVal) {
686     if (aDif > 0)
687       myCompNamesList.append(QString("Comp %1").arg(myCompNamesList.count() + 1));
688     else
689       myCompNamesList.removeLast();
690   }
691
692   AttributeTablesPtr aTablesAttr = myFeature->data()->tables(CollectionPlugin_Field::VALUES_ID());
693   aTablesAttr->setSize(aNbRows, myCompNamesList.size(), myDataTblList.size());
694
695   foreach(QTableWidget* aDataTbl, myDataTblList) {
696     aDataTbl->setColumnCount(theVal + 1);
697     updateHeaders(aDataTbl);
698     for (int i = aOldCol; i < myCompNamesList.count(); i++) {
699       for (int j = 0; j < aNbRows; j++) {
700         aItem = aDataTbl->item(j, i + 1);
701         if (!aItem) {
702           aItem = createDefaultItem();
703           if (j == 0)
704             aItem->setBackgroundColor(Qt::lightGray);
705           aDataTbl->setItem(j, i + 1, aItem);
706         }
707       }
708     }
709   }
710   emit valuesChanged();
711 }
712
713 //**********************************************************************************
714 void CollectionPlugin_WidgetField::onAddStep()
715 {
716   int aMax = myStepSlider->maximum();
717   aMax++;
718   myStepSlider->setMaximum(aMax);
719   myMaxLbl->setText(QString::number(aMax));
720   appendStepControls();
721   myStepSlider->setValue(aMax);
722
723   AttributeTablesPtr aTablesAttr = myFeature->data()->tables(CollectionPlugin_Field::VALUES_ID());
724   aTablesAttr->setSize(aTablesAttr->rows(), aTablesAttr->columns(), myDataTblList.size());
725
726
727   AttributeSelectionListPtr aSelList =
728     myFeature->data()->selectionList(CollectionPlugin_Field::SELECTED_ID());
729   if (!aSelList->isInitialized())
730     return;
731   int aSelNb = aSelList->size();
732   if (aSelNb == 0)
733     return;
734
735   int aColumns = myNbComponentsSpn->value() + 1;
736   int aRows = aSelNb + 1;
737   QTableWidget* aTable = myDataTblList.last();
738   aTable->setRowCount(aRows);
739   QTableWidgetItem* aItem = 0;
740   for(int i = 0; i < aColumns; i++) {
741     if (i == 0) {
742       for(int j = 1; j < aRows; j++) {
743         aItem = aTable->item(j, i);
744         if (!aItem) {
745           aItem = new QTableWidgetItem();
746           aTable->setItem(j, i, aItem);
747         }
748         AttributeSelectionPtr aAttr = aSelList->value(j - 1);
749         aItem->setText(aAttr->namingName().c_str());
750         aItem->setToolTip(aAttr->namingName().c_str());
751       }
752     } else {
753       QString aDefVal = aTable->item(0, i)->text();
754       for(int j = 1; j < aRows; j++) {
755         aItem = aTable->item(j, i);
756         if (!aItem) {
757           aItem = new QTableWidgetItem();
758           aTable->setItem(j, i, aItem);
759         }
760         aItem->setText(aDefVal);
761       }
762     }
763   }
764 }
765
766 //**********************************************************************************
767 void CollectionPlugin_WidgetField::onRemoveStep()
768 {
769   int aMax = myStepSlider->maximum();
770   aMax--;
771   myMaxLbl->setText(QString::number(aMax));
772   removeStepControls();
773   myStepSlider->setMaximum(aMax);
774
775   AttributeTablesPtr aTablesAttr = myFeature->data()->tables(CollectionPlugin_Field::VALUES_ID());
776   aTablesAttr->setSize(aTablesAttr->rows(), aTablesAttr->columns(), myDataTblList.size());
777 }
778
779 //**********************************************************************************
780 void CollectionPlugin_WidgetField::clearData()
781 {
782   foreach(QTableWidget* aDataTbl, myDataTblList) {
783     aDataTbl->setRowCount(1);
784   }
785 }
786
787 //**********************************************************************************
788 void CollectionPlugin_WidgetField::onStepMove(int theStep)
789 {
790   myCurStepLbl->setText(QString::number(theStep));
791   myStepWgt->setCurrentIndex(theStep - 1);
792 }
793
794 //**********************************************************************************
795 bool CollectionPlugin_WidgetField::
796   isValidSelection(const std::shared_ptr<ModuleBase_ViewerPrs>& thePrs)
797 {
798   return (myShapeTypeCombo->currentIndex() == 5)? false : true;
799 }
800
801 //**********************************************************************************
802 void CollectionPlugin_WidgetField::onSelectionChanged()
803 {
804   //if (isEditingMode())
805   //  return;
806
807   if (myActivation) {
808     myActivation = false;
809     return;
810   }
811   // Ignore selection for Parts mode
812   if (myShapeTypeCombo->currentIndex() == 5)
813     return;
814
815   QList<ModuleBase_ViewerPrsPtr> aSelected =
816     myWorkshop->selection()->getSelected(ModuleBase_ISelection::AllControls);
817
818   AttributeSelectionListPtr aSelList =
819     myFeature->data()->selectionList(CollectionPlugin_Field::SELECTED_ID());
820   aSelList->setSelectionType(getSelectionType(myShapeTypeCombo->currentIndex()));
821   aSelList->clear();
822
823   ResultPtr aResult;
824   GeomShapePtr aShape;
825   int aNbData = 0;
826   foreach(ModuleBase_ViewerPrsPtr aPrs, aSelected) {
827     aResult = std::dynamic_pointer_cast<ModelAPI_Result>(aPrs->object());
828     aShape = aPrs->shape();
829     if (!aResult.get() && !aShape.get())
830       continue;
831     if (!aSelList->isInList(aResult, aShape)) {
832       aSelList->append(aResult, aShape);
833       aNbData++;
834     }
835   }
836   int aColumns = myDataTblList.first()->columnCount();
837   int aRows = myDataTblList.first()->rowCount();
838   int aNewRows = aNbData + 1;
839   AttributeTablesPtr aTablesAttr = myFeature->data()->tables(CollectionPlugin_Field::VALUES_ID());
840   aTablesAttr->setSize(aNewRows, aColumns - 1, myDataTblList.size());
841
842   QTableWidgetItem* aItem = 0;
843   foreach(QTableWidget* aTable, myDataTblList) {
844     aTable->setRowCount(aNewRows);
845     if (aNewRows > aRows) {
846       // Add new data
847       for(int i = 0; i < aColumns; i++) {
848         if (i == 0) {
849           for(int j = 1; j < aNewRows; j++) {
850             aItem = aTable->item(j, i);
851             if (!aItem) {
852               aItem = new QTableWidgetItem();
853               aTable->setItem(j, i, aItem);
854             }
855             AttributeSelectionPtr aAttr = aSelList->value(j - 1);
856             aItem->setText(aAttr->namingName().c_str());
857             aItem->setToolTip(aAttr->namingName().c_str());
858           }
859         } else {
860           QString aDefVal = aTable->item(0, i)->text();
861           for(int j = aRows; j < aNewRows; j++) {
862             aItem = aTable->item(j, i);
863             if (!aItem) {
864               aItem = new QTableWidgetItem();
865               aTable->setItem(j, i, aItem);
866             }
867             aItem->setText(aDefVal);
868           }
869         }
870       }
871     } else {
872       // Update only selection name
873       for(int j = 1; j < aNewRows - 1; j++) {
874         AttributeSelectionPtr aAttr = aSelList->value(j);
875         aTable->item(j, 0)->setText(aAttr->namingName().c_str());
876         aTable->item(j, 0)->setToolTip(aAttr->namingName().c_str());
877       }
878     }
879   }
880   emit valuesChanged();
881 }
882
883 //**********************************************************************************
884 void CollectionPlugin_WidgetField::onFieldTypeChanged(int theIdx)
885 {
886   DataTableItemDelegate* aDelegate = 0;
887   aDelegate = dynamic_cast<DataTableItemDelegate*>(myDataTblList.first()->itemDelegate());
888   if (aDelegate) {
889     ModelAPI_AttributeTables::ValueType aOldType = aDelegate->dataType();
890     if (aOldType != theIdx) {
891       aDelegate->setDataType((ModelAPI_AttributeTables::ValueType)theIdx);
892       int aColumns = myDataTblList.first()->columnCount();
893       int aRows = myDataTblList.first()->rowCount();
894       foreach(QTableWidget* aTable, myDataTblList) {
895         for(int i = 1; i < aColumns; i++) {
896           for(int j = 0; j < aRows; j++) {
897             switch (theIdx) {
898             case ModelAPI_AttributeTables::DOUBLE:
899             case ModelAPI_AttributeTables::INTEGER:
900               if ((aOldType == ModelAPI_AttributeTables::BOOLEAN) ||
901                   (aOldType == ModelAPI_AttributeTables::STRING)) {
902                     aTable->item(j, i)->setText("0");
903               }
904               break;
905             case ModelAPI_AttributeTables::BOOLEAN:
906               aTable->item(j, i)->setText(MYFalse);
907               break;
908             case ModelAPI_AttributeTables::STRING:
909               aTable->item(j, i)->setText("");
910               break;
911             }
912           }
913         }
914       }
915       emit valuesChanged();
916     }
917   }
918 }
919
920 //**********************************************************************************
921 void CollectionPlugin_WidgetField::onTableEdited(int theRow, int theCol)
922 {
923   // Do not store here column of names
924   if (theCol == 0)
925     return;
926   if (!myFeature.get())
927     return;
928   QTableWidget* aTable = static_cast<QTableWidget*>(sender());
929   int aNb = myDataTblList.indexOf(aTable);
930   if (aNb == -1)
931     return;
932   ModelAPI_AttributeTables::Value aVal = getValue(aTable->item(theRow, theCol)->text());
933
934   AttributeTablesPtr aTablesAttr = myFeature->data()->tables(CollectionPlugin_Field::VALUES_ID());
935   if (aTablesAttr->isInitialized())
936     aTablesAttr->setValue(aVal,theRow, theCol - 1, aNb);
937   else
938     emit valuesChanged();
939 }
940
941 //**********************************************************************************
942 void CollectionPlugin_WidgetField::onShapeTypeChanged(int theType)
943 {
944   activateSelectionAndFilters(theType == 5? false:true);
945
946   AttributeSelectionListPtr aSelList =
947     myFeature->data()->selectionList(CollectionPlugin_Field::SELECTED_ID());
948
949   std::string aTypeName = getSelectionType(theType);
950   if (aTypeName == aSelList->selectionType())
951     return;
952   aSelList->setSelectionType(aTypeName);
953
954   //Clear old selection
955   clearData();
956   aSelList->clear();
957   AttributeTablesPtr aTablesAttr = myFeature->data()->tables(CollectionPlugin_Field::VALUES_ID());
958   aTablesAttr->setSize(1, myNbComponentsSpn->value(), myDataTblList.size());
959   emit valuesChanged();
960 }
961
962 //**********************************************************************************
963 bool CollectionPlugin_WidgetField::processEnter()
964 {
965   if (myIsTabEdit) {
966     myIsTabEdit = false;
967     return true;
968   }
969   return false;
970 }
971
972 //**********************************************************************************
973 void CollectionPlugin_WidgetField::onFocusChanged(QWidget* theOld, QWidget* theNew)
974 {
975   if (theNew && (!myIsTabEdit))
976     myIsTabEdit = dynamic_cast<QLineEdit*>(theNew);
977 }
978
979 //**********************************************************************************
980 void CollectionPlugin_WidgetField::onRangeChanged(int theMin, int theMax)
981 {
982   myMaxLbl->setText(QString::number(theMax));
983   myRemoveBtn->setEnabled(theMax > 1);
984 }
985
986 //**********************************************************************************
987 void CollectionPlugin_WidgetField::onColumnResize(int theIndex, int theOld, int theNew)
988 {
989   if (myDataTblList.count() < 2)
990     return;
991   QObject* aSender = sender();
992   foreach(QTableWidget* aTable, myDataTblList) {
993     if (aTable->horizontalHeader() != aSender)
994       aTable->setColumnWidth(theIndex, theNew);
995   }
996 }