Salome HOME
Issue #2604: Processing of deleted nodes of sub-objects
[modules/shaper.git] / src / PartSet / PartSet_TreeNodes.cpp
1 // Copyright (C) 2014-2017  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
18 // email : webmaster.salome@opencascade.com<mailto:webmaster.salome@opencascade.com>
19 //
20
21 #include "PartSet_TreeNodes.h"
22
23 #include <ModuleBase_IconFactory.h>
24 #include <ModuleBase_IWorkshop.h>
25
26 #include <PartSetPlugin_Part.h>
27
28 #include <ModelAPI_Session.h>
29 #include <ModelAPI_ResultParameter.h>
30 #include <ModelAPI_ResultField.h>
31 #include <ModelAPI_ResultGroup.h>
32 #include <ModelAPI_ResultConstruction.h>
33 #include <ModelAPI_ResultPart.h>
34 #include <ModelAPI_ResultBody.h>
35 #include <ModelAPI_Tools.h>
36 #include <ModelAPI_ResultBody.h>
37 #include <ModelAPI_CompositeFeature.h>
38 #include <ModelAPI_AttributeDouble.h>
39 #include <ModelAPI_Folder.h>
40 #include <ModelAPI_AttributeReference.h>
41
42 #include <QBrush>
43 #include <QMap>
44
45
46 #define ACTIVE_COLOR QColor(Qt::black)
47 #define SELECTABLE_COLOR QColor(100, 100, 100)
48 #define DISABLED_COLOR QColor(200, 200, 200)
49
50 Qt::ItemFlags aNullFlag;
51 Qt::ItemFlags aDefaultFlag = Qt::ItemIsSelectable | Qt::ItemIsEnabled;
52 Qt::ItemFlags aEditingFlag = Qt::ItemIsSelectable | Qt::ItemIsEnabled | Qt::ItemIsEditable;
53
54
55 ResultPartPtr getPartResult(const ObjectPtr& theObj)
56 {
57   FeaturePtr aFeature = std::dynamic_pointer_cast<ModelAPI_Feature>(theObj);
58   if (aFeature) {
59     ResultPtr aRes = aFeature->firstResult();
60     if (aRes.get() && (aRes->groupName() == ModelAPI_ResultPart::group())) {
61       ResultPartPtr aPartRes = std::dynamic_pointer_cast<ModelAPI_ResultPart>(aRes);
62       // Use only original parts, not a placement results
63       if (aPartRes == aPartRes->original())
64         return aPartRes;
65     }
66   }
67   return ResultPartPtr();
68 }
69
70 bool isCurrentFeature(const ObjectPtr& theObj)
71 {
72   SessionPtr aSession = ModelAPI_Session::get();
73   DocumentPtr aCurDoc = aSession->activeDocument();
74   FeaturePtr aFeature = aCurDoc->currentFeature(true);
75   return aFeature == theObj;
76 }
77
78 //////////////////////////////////////////////////////////////////////////////////
79 QVariant PartSet_TreeNode::data(int theColumn, int theRole) const
80 {
81   if ((theColumn == 1) && (theRole == Qt::ForegroundRole)) {
82     Qt::ItemFlags aFlags = flags(theColumn);
83     if (aFlags == Qt::ItemFlags())
84       return QBrush(DISABLED_COLOR);
85     if (!aFlags.testFlag(Qt::ItemIsEditable))
86       return QBrush(SELECTABLE_COLOR);
87     return ACTIVE_COLOR;
88   }
89   return ModuleBase_ITreeNode::data(theColumn, theRole);
90 }
91
92
93 //////////////////////////////////////////////////////////////////////////////////
94 QVariant PartSet_ObjectNode::data(int theColumn, int theRole) const
95 {
96   switch (theRole) {
97   case Qt::DisplayRole:
98     if (theColumn == 1) {
99       if (myObject->groupName() == ModelAPI_ResultParameter::group()) {
100         ResultParameterPtr aParam = std::dynamic_pointer_cast<ModelAPI_ResultParameter>(myObject);
101         AttributeDoublePtr aValueAttribute =
102           aParam->data()->real(ModelAPI_ResultParameter::VALUE());
103         QString aVal = QString::number(aValueAttribute->value());
104         QString aTitle = QString(myObject->data()->name().c_str());
105         return aTitle + " = " + aVal;
106       }
107       return myObject->data()->name().c_str();
108     }
109     break;
110   case Qt::DecorationRole:
111     switch (theColumn) {
112     case 0:
113       switch (visibilityState()) {
114       case NoneState:
115         return QIcon();
116       case Visible:
117         return QIcon(":pictures/eyeopen.png");
118       case SemiVisible:
119         return QIcon(":pictures/eyemiclosed.png");
120       case Hidden:
121         return QIcon(":pictures/eyeclosed.png");
122       }
123     case 1:
124       if (myObject->groupName() == ModelAPI_Folder::group())
125         return QIcon(":pictures/features_folder.png");
126       else
127         return ModuleBase_IconFactory::get()->getIcon(myObject);
128     case 2:
129       if (isCurrentFeature(myObject))
130         return QIcon(":pictures/arrow.png");
131     }
132   }
133   return PartSet_TreeNode::data(theColumn, theRole);
134 }
135
136 Qt::ItemFlags PartSet_ObjectNode::flags(int theColumn) const
137 {
138   if (myObject->isDisabled()) {
139     return (theColumn == 2) ? Qt::ItemIsSelectable : aNullFlag;
140   } else {
141     DocumentPtr aDoc = myObject->document();
142     SessionPtr aSession = ModelAPI_Session::get();
143     if (aSession->activeDocument() == aDoc)
144       return aEditingFlag;
145   }
146   return aDefaultFlag;
147 }
148
149 PartSet_ObjectNode::VisibilityState PartSet_ObjectNode::visibilityState() const
150 {
151   Qt::ItemFlags aFlags = flags(1);
152   if (aFlags == Qt::ItemFlags())
153     return NoneState;
154
155   if (myObject->groupName() == ModelAPI_ResultParameter::group())
156     return NoneState;
157   ResultPtr aResObj = std::dynamic_pointer_cast<ModelAPI_Result>(myObject);
158   if (aResObj.get()) {
159     ModuleBase_IWorkshop* aWork = workshop();
160     ResultBodyPtr aCompRes = std::dynamic_pointer_cast<ModelAPI_ResultBody>(aResObj);
161     if (aCompRes.get()) {
162       std::list<ResultPtr> aResultsList;
163       ModelAPI_Tools::allSubs(aCompRes, aResultsList);
164       VisibilityState aState = aResultsList.size() == 0 ?
165         (aWork->isVisible(aCompRes) ? Visible : Hidden) : NoneState;
166
167       std::list<ResultPtr>::const_iterator aIt;
168       ResultBodyPtr aCompSub;
169       for (aIt = aResultsList.cbegin(); aIt != aResultsList.cend(); aIt++) {
170         ResultPtr aSubRes = (*aIt);
171         aCompSub = std::dynamic_pointer_cast<ModelAPI_ResultBody>(aSubRes);
172         if (!(aCompSub.get() && aCompSub->numberOfSubs() > 0)) {
173           VisibilityState aS = aWork->isVisible(aSubRes) ? Visible : Hidden;
174           if (aState == NoneState)
175             aState = aS;
176           else if (aState != aS) {
177             aState = SemiVisible;
178             break;
179           }
180         }
181       }
182       return aState;
183     } else {
184       if (aWork->isVisible(aResObj))
185         return Visible;
186       else
187         return Hidden;
188     }
189   }
190   return NoneState;
191 }
192
193 int PartSet_ObjectNode::numberOfSubs() const
194 {
195   ResultBodyPtr aCompRes = std::dynamic_pointer_cast<ModelAPI_ResultBody>(myObject);
196   if (aCompRes.get())
197    return aCompRes->numberOfSubs(true);
198   else {
199     CompositeFeaturePtr aCompFeature =
200       std::dynamic_pointer_cast<ModelAPI_CompositeFeature>(myObject);
201     if (aCompFeature.get() && aCompFeature->data()->isValid())
202       return aCompFeature->numberOfSubs(true);
203     else {
204       ResultFieldPtr aFieldRes = std::dynamic_pointer_cast<ModelAPI_ResultField>(myObject);
205       if (aFieldRes.get())
206         return aFieldRes->stepsSize();
207     }
208   }
209   return 0;
210 }
211
212
213 ObjectPtr PartSet_ObjectNode::subObject(int theId) const
214 {
215   ResultBodyPtr aCompRes = std::dynamic_pointer_cast<ModelAPI_ResultBody>(myObject);
216   if (aCompRes.get())
217     return aCompRes->subResult(theId, true);
218   else {
219     CompositeFeaturePtr aCompFeature =
220       std::dynamic_pointer_cast<ModelAPI_CompositeFeature>(myObject);
221     if (aCompFeature.get())
222       return aCompFeature->subFeature(theId, true);
223   }
224   return ObjectPtr();
225 }
226
227 void PartSet_ObjectNode::update()
228 {
229   int aNb = numberOfSubs();
230   if (aNb > 0) {
231     ResultFieldPtr aFieldRes = std::dynamic_pointer_cast<ModelAPI_ResultField>(myObject);
232
233     // If the object is a field result then delete extra sub-objects
234     if (aFieldRes.get()) {
235       while (myChildren.size() > aNb) {
236         ModuleBase_ITreeNode* aNode = myChildren.last();
237         myChildren.removeAll(aNode);
238         delete aNode;
239       }
240     }
241     else {
242       ObjectPtr aObj;
243       ModuleBase_ITreeNode* aNode;
244       int aId = 0;
245       while (aId < myChildren.size()) {
246         aNode = myChildren.at(aId);
247         aObj = subObject(aId);
248         if (aNode->object() != aObj) {
249           myChildren.removeAll(aNode);
250           delete aNode;
251         }
252         else
253           aId++;
254       }
255     }
256
257     ModuleBase_ITreeNode* aNode;
258     ObjectPtr aBody;
259     int i;
260     for (i = 0; i < aNb; i++) {
261       aBody = subObject(i);
262       if (aBody.get()) {
263         if (i < myChildren.size()) {
264           aNode = myChildren.at(i);
265           if (aNode->object() != aBody) {
266             ((PartSet_ObjectNode*)aNode)->setObject(aBody);
267           }
268         }
269         else {
270           aNode = new PartSet_ObjectNode(aBody, this);
271           myChildren.append(aNode);
272         }
273       }
274       else if (aFieldRes.get()) {
275         ModelAPI_ResultField::ModelAPI_FieldStep* aStep = aFieldRes->step(i);
276         if (i < myChildren.size()) {
277           PartSet_StepNode* aStepNode = static_cast<PartSet_StepNode*>(myChildren.at(i));
278           if (aStepNode->entity() != aStep) {
279             aStepNode->setEntity(aStep);
280           }
281         }
282         else {
283           aNode = new PartSet_StepNode(aStep, this);
284           myChildren.append(aNode);
285         }
286       }
287     }
288     // Delete extra objects
289     while (myChildren.size() > aNb) {
290       aNode = myChildren.takeLast();
291       delete aNode;
292     }
293     foreach(ModuleBase_ITreeNode* aNode, myChildren) {
294       aNode->update();
295     }
296   }
297   else {
298     deleteChildren();
299   }
300 }
301
302 QTreeNodesList PartSet_ObjectNode::objectCreated(const QObjectPtrList& theObjects)
303 {
304   QTreeNodesList aResult;
305   int aNb = numberOfSubs();
306   if (aNb > 0) {
307     ModuleBase_ITreeNode* aNode;
308     ResultFieldPtr aFieldRes = std::dynamic_pointer_cast<ModelAPI_ResultField>(myObject);
309     ObjectPtr aBody;
310     int i;
311     for (i = 0; i < aNb; i++) {
312       aBody = subObject(i);
313       if (aBody.get()) {
314         if (i < myChildren.size()) {
315           aNode = myChildren.at(i);
316           if (aNode->object() != aBody) {
317             ((PartSet_ObjectNode*)aNode)->setObject(aBody);
318             aResult.append(aNode);
319           }
320         }
321         else {
322           aNode = new PartSet_ObjectNode(aBody, this);
323           myChildren.append(aNode);
324           aResult.append(aNode);
325           aNode->update();
326         }
327       }
328       else {
329         ModelAPI_ResultField::ModelAPI_FieldStep* aStep = aFieldRes->step(i);
330         if (i < myChildren.size()) {
331           PartSet_StepNode* aStepNode = static_cast<PartSet_StepNode*>(myChildren.at(i));
332           if (aStepNode->entity() != aStep) {
333             aStepNode->setEntity(aStep);
334           }
335         }
336         else {
337           aNode = new PartSet_StepNode(aStep, this);
338           myChildren.append(aNode);
339         }
340       }
341     }
342     foreach(ModuleBase_ITreeNode* aNode, myChildren) {
343       aResult.append(aNode->objectCreated(theObjects));
344     }
345   }
346   return aResult;
347 }
348
349 QTreeNodesList PartSet_ObjectNode::objectsDeleted(
350   const DocumentPtr& theDoc, const QString& theGroup)
351 {
352   QTreeNodesList aResult;
353   int aNb = numberOfSubs();
354   if (aNb != myChildren.size()) {
355     if (aNb == 0) {
356       deleteChildren();
357       aResult.append(this);
358     }
359     else {
360       // Delete extra objects
361       bool isDeleted = false;
362       ObjectPtr aObj;
363       ModuleBase_ITreeNode* aNode;
364       int aId = 0;
365       while (aId < myChildren.size()) {
366         aNode = myChildren.at(aId);
367         aObj = subObject(aId);
368         if (aNode->object() != aObj) {
369           myChildren.removeAll(aNode);
370           delete aNode;
371           isDeleted = true;
372         }
373         else
374           aId++;
375       }
376       if (isDeleted)
377         aResult.append(this);
378       int i = 0;
379       ObjectPtr aBody;
380       foreach(ModuleBase_ITreeNode* aNode, myChildren) {
381         aBody = subObject(i);
382         ((PartSet_ObjectNode*)aNode)->setObject(aBody);
383         aResult.append(aNode->objectsDeleted(theDoc, theGroup));
384         i++;
385       }
386     }
387   }
388   return aResult;
389 }
390 //////////////////////////////////////////////////////////////////////////////////
391 PartSet_FolderNode::PartSet_FolderNode(ModuleBase_ITreeNode* theParent,
392   FolderType theType)
393   : PartSet_TreeNode(theParent), myType(theType)
394 {
395 }
396
397 QString PartSet_FolderNode::name() const
398 {
399   switch (myType) {
400   case ParametersFolder:
401     return QObject::tr("Parameters");
402   case ConstructionFolder:
403     return QObject::tr("Constructions");
404   case PartsFolder:
405     return QObject::tr("Parts");
406   case ResultsFolder:
407     return QObject::tr("Results");
408   case FieldsFolder:
409     return QObject::tr("Fields");
410   case GroupsFolder:
411     return QObject::tr("Groups");
412   }
413   return "NoName";
414 }
415
416
417 QVariant PartSet_FolderNode::data(int theColumn, int theRole) const
418 {
419   static QIcon aParamsIco(":pictures/params_folder.png");
420   static QIcon aConstrIco(":pictures/constr_folder.png");
421
422   if (theColumn == 1) {
423     switch (theRole) {
424     case Qt::DisplayRole:
425       return name() + QString(" (%1)").arg(childrenCount());
426     case Qt::DecorationRole:
427       switch (myType) {
428       case ParametersFolder:
429         return aParamsIco;
430       case ConstructionFolder:
431         return aConstrIco;
432       case PartsFolder:
433         return aConstrIco;
434       case ResultsFolder:
435         return aConstrIco;
436       case FieldsFolder:
437         return aConstrIco;
438       case GroupsFolder:
439         return aConstrIco;
440       }
441     }
442   }
443   return PartSet_TreeNode::data(theColumn, theRole);
444 }
445
446 Qt::ItemFlags PartSet_FolderNode::flags(int theColumn) const
447 {
448   SessionPtr aSession = ModelAPI_Session::get();
449   DocumentPtr aActiveDoc = aSession->activeDocument();
450   if (theColumn == 1) {
451     if (document() == aActiveDoc)
452       return aEditingFlag;
453   }
454   return aDefaultFlag;
455 }
456
457 ModuleBase_ITreeNode* PartSet_FolderNode::createNode(const ObjectPtr& theObj)
458 {
459   //ResultCompSolidPtr aCompRes = std::dynamic_pointer_cast<ModelAPI_ResultCompSolid>(theObj);
460   //if (aCompRes.get())
461   //  return new PartSet_CompsolidNode(theObj, this);
462   return new PartSet_ObjectNode(theObj, this);
463 }
464
465 void PartSet_FolderNode::update()
466 {
467   DocumentPtr aDoc = document();
468   if (!aDoc.get())
469     return;
470
471   // Remove extra sub-nodes
472   int aIndex;
473   int aId = 0;
474   while (aId < myChildren.size()) {
475     ModuleBase_ITreeNode* aNode = myChildren.at(aId);
476     aIndex = aDoc->index(aNode->object(), true);
477     if ((aIndex == -1) || (aId != aIndex)) {
478       myChildren.removeAll(aNode);
479       delete aNode;
480     } else
481       aId++;
482   }
483
484   // Add new nodes
485   std::string aGroup = groupName();
486   int aSize = aDoc->size(aGroup, true);
487   for (int i = 0; i < aSize; i++) {
488     ObjectPtr aObj = aDoc->object(aGroup, i, true);
489     if (i < myChildren.size()) {
490       if (myChildren.at(i)->object() != aObj) {
491         ModuleBase_ITreeNode* aNode = createNode(aObj);
492         myChildren.insert(i, aNode);
493       }
494     } else {
495       ModuleBase_ITreeNode* aNode = createNode(aObj);
496       myChildren.append(aNode);
497     }
498   }
499
500   foreach(ModuleBase_ITreeNode* aNode, myChildren) {
501     aNode->update();
502   }
503 }
504
505 std::string PartSet_FolderNode::groupName() const
506 {
507   switch (myType) {
508   case ParametersFolder:
509     return ModelAPI_ResultParameter::group();
510   case ConstructionFolder:
511     return ModelAPI_ResultConstruction::group();
512   case PartsFolder:
513     return ModelAPI_ResultPart::group();
514   case ResultsFolder:
515     return ModelAPI_ResultBody::group();
516   case FieldsFolder:
517     return ModelAPI_ResultField::group();
518   case GroupsFolder:
519     return ModelAPI_ResultGroup::group();
520   }
521   return "";
522 }
523
524 QTreeNodesList PartSet_FolderNode::objectCreated(const QObjectPtrList& theObjects)
525 {
526   QTreeNodesList aResult;
527   std::string aName = groupName();
528   DocumentPtr aDoc = document();
529   int aIdx = -1;
530   QMap<int, ModuleBase_ITreeNode*> aNewNodes;
531   foreach(ObjectPtr aObj, theObjects) {
532     if ((aObj->document() == aDoc) && (aObj->groupName() == aName)) {
533       aIdx = aDoc->index(aObj, true);
534       if (aIdx != -1) {
535         bool aHasObject = (aIdx < myChildren.size()) && (myChildren.at(aIdx)->object() == aObj);
536         if (!aHasObject) {
537           ModuleBase_ITreeNode* aNode = createNode(aObj);
538           aNewNodes[aIdx] = aNode;
539           aResult.append(aNode);
540           aNode->update();
541         }
542       }
543     }
544   }
545   // Add nodes in correct order
546   int i;
547   for (i = 0; i < myChildren.size(); i++) {
548     if (aNewNodes.contains(i)) {
549       myChildren.insert(i, aNewNodes[i]);
550       aNewNodes.remove(i);
551     }
552   }
553   while (aNewNodes.size()) {
554     i = myChildren.size();
555     myChildren.append(aNewNodes[i]);
556     aNewNodes.remove(i);
557   }
558   foreach(ModuleBase_ITreeNode* aNode, myChildren) {
559     aResult.append(aNode->objectCreated(theObjects));
560   }
561   return aResult;
562 }
563
564 QTreeNodesList PartSet_FolderNode::objectsDeleted(const DocumentPtr& theDoc,
565   const QString& theGroup)
566 {
567   DocumentPtr aDoc = document();
568   QTreeNodesList aResult;
569   if ((theGroup.toStdString() == groupName()) && (theDoc == aDoc)) {
570     QTreeNodesList aDelList;
571     int aIndex;
572     int aId = 0;
573     bool aRemoved = false;
574     while (aId < myChildren.size()) {
575       ModuleBase_ITreeNode* aNode = myChildren.at(aId);
576       aIndex = aDoc->index(aNode->object(), true);
577       if ((aIndex == -1) || (aId != aIndex)) {
578         myChildren.removeAll(aNode);
579         delete aNode;
580         aRemoved = true;
581       }  else
582         aId++;
583     }
584     if (aRemoved)
585       aResult.append(this);
586
587     foreach(ModuleBase_ITreeNode* aNode, myChildren) {
588       aResult.append(aNode->objectsDeleted(theDoc, theGroup));
589     }
590   }
591   return aResult;
592 }
593
594 //////////////////////////////////////////////////////////////////////////////////
595 QTreeNodesList PartSet_FeatureFolderNode::objectCreated(const QObjectPtrList& theObjects)
596 {
597   QTreeNodesList aResult;
598   // Process the root sub-objects
599   DocumentPtr aDoc = document();
600   int aIdx = -1;
601   int aNb = numberOfFolders();
602   QMap<int, ModuleBase_ITreeNode*> aNewNodes;
603   foreach(ObjectPtr aObj, theObjects) {
604     if (aDoc == aObj->document()) {
605       if ((aObj->groupName() == ModelAPI_Feature::group()) ||
606         (aObj->groupName() == ModelAPI_Folder::group())){
607         aIdx = aDoc->index(aObj, true);
608         if (aIdx != -1) {
609           ModuleBase_ITreeNode* aNode = createNode(aObj);
610           aIdx += aNb;
611           bool aHasObject = (aIdx < myChildren.size()) && (myChildren.at(aIdx)->object() == aObj);
612           if (!aHasObject) {
613             aNewNodes[aIdx] = aNode;
614             aResult.append(aNode);
615             aNode->update();
616           }
617         }
618       }
619     }
620   }
621   // To add in correct order
622   int i;
623   for (i = 0; i < myChildren.size(); i++) {
624     if (aNewNodes.contains(i)) {
625       myChildren.insert(i, aNewNodes[i]);
626       aNewNodes.remove(i);
627     }
628   }
629   while (aNewNodes.size()) {
630     i = myChildren.size();
631     myChildren.append(aNewNodes[i]);
632     aNewNodes.remove(i);
633   }
634
635   // Update sub-folders
636   foreach(ModuleBase_ITreeNode* aNode, myChildren) {
637     aResult.append(aNode->objectCreated(theObjects));
638   }
639   return aResult;
640 }
641
642 QTreeNodesList PartSet_FeatureFolderNode::objectsDeleted(const DocumentPtr& theDoc,
643   const QString& theGroup)
644 {
645   QTreeNodesList aResult;
646
647   // Process sub-folders
648   foreach(ModuleBase_ITreeNode* aNode, myChildren) {
649     if (aNode->childrenCount() > 0) { // aFolder node
650       QTreeNodesList aList = aNode->objectsDeleted(theDoc, theGroup);
651       if (aList.size() > 0)
652         aResult.append(aList);
653     }
654   }
655
656   // Process root
657   DocumentPtr aDoc = document();
658   int aNb = numberOfFolders();
659   bool isGroup = ((theGroup.toStdString() == ModelAPI_Feature::group()) ||
660     (theGroup.toStdString() == ModelAPI_Folder::group()));
661   if ((theDoc == aDoc) && isGroup) {
662     int aIndex;
663     int aId = 0;
664     bool aRemoved = false;
665     while (aId < myChildren.size()) {
666       ModuleBase_ITreeNode* aNode = myChildren.at(aId);
667       if (aNode->object().get()) {
668         aIndex = aDoc->index(aNode->object(), true);
669         if ((aIndex == -1) || (aId != (aIndex + aNb))) {
670           myChildren.removeAll(aNode);
671           delete aNode;
672           aRemoved = true;
673           continue;
674         }
675       }
676       aId++;
677     }
678     if (aRemoved)
679       aResult.append(this);
680   }
681   return aResult;
682 }
683
684 ModuleBase_ITreeNode* PartSet_FeatureFolderNode::findParent(const DocumentPtr& theDoc,
685   QString theGroup)
686 {
687   ModuleBase_ITreeNode* aResult = 0;
688   foreach(ModuleBase_ITreeNode* aNode, myChildren) {
689     aResult = aNode->findParent(theDoc, theGroup);
690     if (aResult) {
691       return aResult;
692     }
693   }
694   bool isGroup = ((theGroup.toStdString() == ModelAPI_Feature::group()) ||
695     (theGroup.toStdString() == ModelAPI_Folder::group()));
696   if ((theDoc == document()) && isGroup)
697     return this;
698   return 0;
699 }
700
701
702 //////////////////////////////////////////////////////////////////////////////////
703 PartSet_RootNode::PartSet_RootNode() : PartSet_FeatureFolderNode(0), myWorkshop(0)
704 {
705   SessionPtr aSession = ModelAPI_Session::get();
706   DocumentPtr aDoc = aSession->moduleDocument();
707
708   myParamsFolder = new PartSet_FolderNode(this, PartSet_FolderNode::ParametersFolder);
709   myConstrFolder = new PartSet_FolderNode(this, PartSet_FolderNode::ConstructionFolder);
710   myPartsFolder = new PartSet_FolderNode(this, PartSet_FolderNode::PartsFolder);
711
712   myChildren.append(myParamsFolder);
713   myChildren.append(myConstrFolder);
714   myChildren.append(myPartsFolder);
715
716   update();
717 }
718
719
720 void PartSet_RootNode::update()
721 {
722   myParamsFolder->update();
723   myConstrFolder->update();
724   myPartsFolder->update();
725
726   // Update features content
727   DocumentPtr aDoc = document();
728   int aNb = numberOfFolders();
729
730   // Remove extra sub-nodes
731   int aIndex;
732   int aId = 0;
733   while (aId < myChildren.size()) {
734     ModuleBase_ITreeNode* aNode = myChildren.at(aId);
735     if (aNode->object().get()) {
736       aIndex = aDoc->index(aNode->object(), true);
737       if ((aIndex == -1) || (aId != (aIndex + aNb))) {
738         myChildren.removeAll(aNode);
739         delete aNode;
740         continue;
741       }
742     }
743     aId++;
744   }
745
746   // Add new nodes
747   std::string aGroup = ModelAPI_Feature::group();
748   int aSize = aDoc->size(aGroup, true);
749   FeaturePtr aFeature;
750   for (int i = 0; i < aSize; i++) {
751     ObjectPtr aObj = aDoc->object(aGroup, i, true);
752     aId = i + aNb; // Take into account existing folders
753     if (aId < myChildren.size()) {
754       if (myChildren.at(aId)->object() != aObj) {
755         ModuleBase_ITreeNode* aNode = createNode(aObj);
756         myChildren.insert(aId, aNode);
757       }
758     } else {
759       ModuleBase_ITreeNode* aNode = createNode(aObj);
760       myChildren.append(aNode);
761     }
762   }
763   // Update sub-folders
764   foreach(ModuleBase_ITreeNode* aNode, myChildren) {
765     if ((aNode->type() == PartSet_ObjectFolderNode::typeId()) ||
766       (aNode->type() == PartSet_PartRootNode::typeId()))
767       aNode->update();
768   }
769 }
770
771 DocumentPtr PartSet_RootNode::document() const
772 {
773   return ModelAPI_Session::get()->moduleDocument();
774 }
775
776 ModuleBase_ITreeNode* PartSet_RootNode::createNode(const ObjectPtr& theObj)
777 {
778   if (theObj->groupName() == ModelAPI_Folder::group())
779     return new PartSet_ObjectFolderNode(theObj, this);
780
781   FeaturePtr aFeature = std::dynamic_pointer_cast<ModelAPI_Feature>(theObj);
782   if (aFeature->getKind() == PartSetPlugin_Part::ID())
783     return new PartSet_PartRootNode(theObj, this);
784
785   return new PartSet_ObjectNode(theObj, this);
786 }
787
788 //////////////////////////////////////////////////////////////////////////////////
789 PartSet_PartRootNode::PartSet_PartRootNode(const ObjectPtr& theObj, ModuleBase_ITreeNode* theParent)
790   : PartSet_FeatureFolderNode(theParent), myObject(theObj)
791 {
792   myParamsFolder = new PartSet_FolderNode(this, PartSet_FolderNode::ParametersFolder);
793   myConstrFolder = new PartSet_FolderNode(this, PartSet_FolderNode::ConstructionFolder);
794   myResultsFolder = new PartSet_FolderNode(this, PartSet_FolderNode::ResultsFolder);
795   myFieldsFolder = new PartSet_FolderNode(this, PartSet_FolderNode::FieldsFolder);
796   myGroupsFolder = new PartSet_FolderNode(this, PartSet_FolderNode::GroupsFolder);
797
798   myChildren.append(myParamsFolder);
799   myChildren.append(myConstrFolder);
800   myChildren.append(myResultsFolder);
801
802   update();
803 }
804
805 void PartSet_PartRootNode::deleteChildren()
806 {
807   if (!myFieldsFolder->childrenCount()) {
808     delete myFieldsFolder;
809   }
810   if (!myGroupsFolder->childrenCount()) {
811     delete myGroupsFolder;
812   }
813   PartSet_FeatureFolderNode::deleteChildren();
814 }
815
816
817 void PartSet_PartRootNode::update()
818 {
819   DocumentPtr aDoc = document();
820   if (!aDoc.get())
821     return;
822
823   myParamsFolder->update();
824   myConstrFolder->update();
825   myResultsFolder->update();
826   myFieldsFolder->update();
827   myGroupsFolder->update();
828
829   bool aHasFields = myFieldsFolder->childrenCount() > 0;
830   bool aHasGroups = myGroupsFolder->childrenCount() > 0;
831   if (aHasFields && (!myChildren.contains(myFieldsFolder))) {
832     myChildren.insert(3, myFieldsFolder);
833   }
834   if (aHasGroups && (!myChildren.contains(myGroupsFolder))) {
835     myChildren.insert(aHasFields ? 4 : 3, myGroupsFolder);
836   }
837
838   // Update features content
839   int aRows = numberOfFolders();
840
841   // Remove extra sub-nodes
842   int aIndex = -1;
843   int aId = 0;
844   while (aId < myChildren.size()) {
845     ModuleBase_ITreeNode* aNode = myChildren.at(aId);
846     if (aNode->object().get()) {
847       aIndex = aDoc->index(aNode->object(), true);
848       if ((aIndex == -1) || (aId != (aIndex + aRows))) {
849         myChildren.removeAll(aNode);
850         delete aNode;
851         continue;
852       }
853     }
854     aId++;
855   }
856
857   std::string aGroup = ModelAPI_Feature::group();
858   int aSize = aDoc->size(aGroup, true);
859   FeaturePtr aFeature;
860   for (int i = 0; i < aSize; i++) {
861     ObjectPtr aObj = aDoc->object(aGroup, i, true);
862     aId = i + aRows; // Take into account existing folders
863     if (aId < myChildren.size()) {
864       if (myChildren.at(aId)->object() != aObj) {
865         ModuleBase_ITreeNode* aNode = createNode(aObj);
866         myChildren.insert(aId, aNode);
867       }
868     } else {
869       ModuleBase_ITreeNode* aNode = createNode(aObj);
870       myChildren.append(aNode);
871     }
872   }
873   // Update sub-folders
874   foreach(ModuleBase_ITreeNode* aNode, myChildren) {
875     if (aNode->type() == PartSet_ObjectFolderNode::typeId())
876       aNode->update();
877   }
878 }
879
880 DocumentPtr PartSet_PartRootNode::document() const
881 {
882   ResultPartPtr aPartRes = getPartResult(myObject);
883   if (aPartRes.get())
884     return aPartRes->partDoc();
885   return DocumentPtr();
886 }
887
888 QVariant PartSet_PartRootNode::data(int theColumn, int theRole) const
889 {
890   switch (theColumn) {
891   case 1:
892     switch (theRole) {
893     case Qt::DisplayRole:
894       return QString(myObject->data()->name().c_str());
895     case Qt::DecorationRole:
896       return ModuleBase_IconFactory::get()->getIcon(myObject);
897     }
898   case 2:
899     if (theRole == Qt::DecorationRole)
900       if (isCurrentFeature(myObject))
901         return QIcon(":pictures/arrow.png");
902   }
903   return PartSet_TreeNode::data(theColumn, theRole);
904 }
905
906 Qt::ItemFlags PartSet_PartRootNode::flags(int theColumn) const
907 {
908   SessionPtr aSession = ModelAPI_Session::get();
909   DocumentPtr aActiveDoc = aSession->activeDocument();
910   if ((aActiveDoc == document()) || (myObject->document() == aActiveDoc))
911     return aEditingFlag;
912   return aDefaultFlag;
913 }
914
915 ModuleBase_ITreeNode* PartSet_PartRootNode::createNode(const ObjectPtr& theObj)
916 {
917   if (theObj->groupName() == ModelAPI_Folder::group())
918     return new PartSet_ObjectFolderNode(theObj, this);
919   return new PartSet_ObjectNode(theObj, this);
920 }
921
922 int PartSet_PartRootNode::numberOfFolders() const
923 {
924   int aNb = 3;
925   if (myFieldsFolder->childrenCount() > 0)
926     aNb++;
927   if (myGroupsFolder->childrenCount() > 0)
928     aNb++;
929   return aNb;
930 }
931
932 QTreeNodesList PartSet_PartRootNode::objectCreated(const QObjectPtrList& theObjects)
933 {
934   QTreeNodesList aResult = PartSet_FeatureFolderNode::objectCreated(theObjects);
935   if (!myFieldsFolder->childrenCount()) {
936     QTreeNodesList aList = myFieldsFolder->objectCreated(theObjects);
937     if (aList.size()) {
938       myChildren.insert(3, myFieldsFolder);
939       aResult.append(myFieldsFolder);
940       aResult.append(aList);
941     }
942   }
943   if (!myGroupsFolder->childrenCount()) {
944     QTreeNodesList aList = myGroupsFolder->objectCreated(theObjects);
945     if (aList.size()) {
946       myChildren.insert(myFieldsFolder->childrenCount()? 4 : 3, myGroupsFolder);
947       aResult.append(myGroupsFolder);
948       aResult.append(aList);
949     }
950   }
951   return aResult;
952 }
953
954 QTreeNodesList PartSet_PartRootNode::objectsDeleted(const DocumentPtr& theDoc,
955   const QString& theGroup)
956 {
957   QTreeNodesList aResult;
958   if (myFieldsFolder->childrenCount()) {
959     QTreeNodesList aList = myFieldsFolder->objectsDeleted(theDoc, theGroup);
960     if (aList.size()) {
961       aResult.append(aList);
962       if (!myFieldsFolder->childrenCount())
963         myChildren.removeAll(myFieldsFolder);
964     }
965   }
966   if (myGroupsFolder->childrenCount()) {
967     QTreeNodesList aList = myGroupsFolder->objectsDeleted(theDoc, theGroup);
968     if (aList.size()) {
969       aResult.append(aList);
970       if (!myGroupsFolder->childrenCount())
971         myChildren.removeAll(myGroupsFolder);
972     }
973   }
974   aResult.append(PartSet_FeatureFolderNode::objectsDeleted(theDoc, theGroup));
975   return aResult;
976 }
977
978 //////////////////////////////////////////////////////////////////////////////////
979 void PartSet_ObjectFolderNode::update()
980 {
981   int aFirst, aLast;
982   getFirstAndLastIndex(aFirst, aLast);
983   if ((aFirst == -1) || (aLast == -1)) {
984     deleteChildren();
985     return;
986   }
987
988   int aNbItems = aLast - aFirst + 1;
989   if (!aNbItems) {
990     deleteChildren();
991     return;
992   }
993
994   DocumentPtr aDoc = myObject->document();
995   // Delete obsolete nodes
996   int aId = 0;
997   while (aId < myChildren.size()) {
998     ModuleBase_ITreeNode* aNode = myChildren.at(aId);
999     if ((aFirst + aId) < aDoc->size(ModelAPI_Feature::group(), true)) {
1000       if (aNode->object() != aDoc->object(ModelAPI_Feature::group(), aFirst + aId)) {
1001         myChildren.removeAll(aNode);
1002         delete aNode;
1003         continue;
1004       }
1005     }
1006     aId++;
1007   }
1008
1009   // Add new nodes
1010   ModuleBase_ITreeNode* aNode;
1011   for (int i = 0; i < aNbItems; i++) {
1012     ObjectPtr aObj = aDoc->object(ModelAPI_Feature::group(), aFirst + i);
1013     if (i < myChildren.size()) {
1014       if (aObj != myChildren.at(i)->object()) {
1015         aNode = new PartSet_ObjectNode(aObj, this);
1016         myChildren.insert(i, aNode);
1017       }
1018     } else {
1019       aNode = new PartSet_ObjectNode(aObj, this);
1020       myChildren.append(aNode);
1021     }
1022   }
1023 }
1024
1025 QTreeNodesList PartSet_ObjectFolderNode::objectCreated(const QObjectPtrList& theObjects)
1026 {
1027   QTreeNodesList aResult;
1028   int aFirst, aLast;
1029   getFirstAndLastIndex(aFirst, aLast);
1030   if ((aFirst == -1) || (aLast == -1)) {
1031     return aResult;
1032   }
1033   int aNbItems = aLast - aFirst + 1;
1034   if (!aNbItems) {
1035     return aResult;
1036   }
1037   DocumentPtr aDoc = myObject->document();
1038   // Add new nodes
1039   ModuleBase_ITreeNode* aNode;
1040   for (int i = 0; i < aNbItems; i++) {
1041     ObjectPtr aObj = aDoc->object(ModelAPI_Feature::group(), aFirst + i);
1042     if (i < myChildren.size()) {
1043       if (aObj != myChildren.at(i)->object()) {
1044         aNode = new PartSet_ObjectNode(aObj, this);
1045         myChildren.insert(i, aNode);
1046         aResult.append(aNode);
1047       }
1048     } else {
1049       aNode = new PartSet_ObjectNode(aObj, this);
1050       myChildren.append(aNode);
1051       aResult.append(aNode);
1052     }
1053   }
1054   return aResult;
1055 }
1056
1057 QTreeNodesList PartSet_ObjectFolderNode::objectsDeleted(const DocumentPtr& theDoc,
1058   const QString& theGroup)
1059 {
1060   QTreeNodesList aResult;
1061   int aFirst, aLast;
1062   getFirstAndLastIndex(aFirst, aLast);
1063   if ((aFirst == -1) || (aLast == -1)) {
1064     return aResult;
1065   }
1066   int aNbItems = aLast - aFirst + 1;
1067   if (!aNbItems) {
1068     return aResult;
1069   }
1070   DocumentPtr aDoc = myObject->document();
1071   // Delete obsolete nodes
1072   bool aRemoved = false;
1073   int aId = 0;
1074   while (aId < myChildren.size()) {
1075     ModuleBase_ITreeNode* aNode = myChildren.at(aId);
1076     if ((aFirst + aId) < aDoc->size(ModelAPI_Feature::group(), true)) {
1077       if (aNode->object() != aDoc->object(ModelAPI_Feature::group(), aFirst + aId)) {
1078         myChildren.removeAll(aNode);
1079         delete aNode;
1080         aRemoved = true;
1081         continue;
1082       }
1083     }
1084     aId++;
1085   }
1086   if (aRemoved) {
1087     aResult.append(this);
1088   }
1089   return aResult;
1090 }
1091
1092 FeaturePtr PartSet_ObjectFolderNode::getFeature(const std::string& theId) const
1093 {
1094   FolderPtr aFolder = std::dynamic_pointer_cast<ModelAPI_Folder>(myObject);
1095   AttributeReferencePtr aFeatAttr = aFolder->data()->reference(theId);
1096   if (aFeatAttr)
1097     return ModelAPI_Feature::feature(aFeatAttr->value());
1098   return FeaturePtr();
1099 }
1100
1101 void PartSet_ObjectFolderNode::getFirstAndLastIndex(int& theFirst, int& theLast) const
1102 {
1103   DocumentPtr aDoc = myObject->document();
1104   FolderPtr aFolder = std::dynamic_pointer_cast<ModelAPI_Folder>(myObject);
1105
1106   FeaturePtr aFirstFeatureInFolder = getFeature(ModelAPI_Folder::FIRST_FEATURE_ID());
1107   if (!aFirstFeatureInFolder.get()) {
1108     theFirst = -1;
1109     return;
1110   }
1111   FeaturePtr aLastFeatureInFolder = getFeature(ModelAPI_Folder::LAST_FEATURE_ID());
1112   if (!aLastFeatureInFolder.get()) {
1113     theLast = -1;
1114     return;
1115   }
1116
1117   theFirst = aDoc->index(aFirstFeatureInFolder);
1118   theLast = aDoc->index(aLastFeatureInFolder);
1119 }
1120
1121
1122 //////////////////////////////////////////////////////////////////////////////////
1123 QVariant PartSet_StepNode::data(int theColumn, int theRole) const
1124 {
1125   if ((theColumn == 1) && (theRole == Qt::DisplayRole)) {
1126     ModelAPI_ResultField::ModelAPI_FieldStep* aStep =
1127       dynamic_cast<ModelAPI_ResultField::ModelAPI_FieldStep*>(myEntity);
1128
1129     return "Step " + QString::number(aStep->id() + 1) + " " +
1130       aStep->field()->textLine(aStep->id()).c_str();
1131   }
1132   return PartSet_TreeNode::data(theColumn, theRole);
1133 }