X-Git-Url: http://git.salome-platform.org/gitweb/?a=blobdiff_plain;ds=sidebyside;f=src%2FModel%2FModel_AttributeSelectionList.cpp;h=b44767750922a7ace3dc5ff37bdf1c35ece593ed;hb=061a63480f6840b6d945f7744b3b972e2d4cb25d;hp=8191553d15d588e3f0c1262a0aab7e07863e1b3b;hpb=56b43fa7eef0b09ec2e4c12532cf21e2021ca534;p=modules%2Fshaper.git diff --git a/src/Model/Model_AttributeSelectionList.cpp b/src/Model/Model_AttributeSelectionList.cpp index 8191553d1..b44767750 100644 --- a/src/Model/Model_AttributeSelectionList.cpp +++ b/src/Model/Model_AttributeSelectionList.cpp @@ -10,7 +10,9 @@ #include "Model_Events.h" #include "Model_Data.h" +#include #include +#include #include #include @@ -21,7 +23,8 @@ using namespace std; void Model_AttributeSelectionList::append( - const ResultPtr& theContext, const std::shared_ptr& theSubShape) + const ResultPtr& theContext, const std::shared_ptr& theSubShape, + const bool theTemporarily) { // do not use the degenerated edge as a shape, a list is not incremented in this case if (theSubShape.get() && !theSubShape->isNull() && theSubShape->isEdge()) { @@ -41,11 +44,16 @@ void Model_AttributeSelectionList::append( } aNewAttr->setID(id()); mySize->Set(aNewTag); - aNewAttr->setValue(theContext, theSubShape); + aNewAttr->setValue(theContext, theSubShape, theTemporarily); + if (theTemporarily) + myTmpAttr = aNewAttr; + else + myTmpAttr.reset(); owner()->data()->sendAttributeUpdated(this); } -void Model_AttributeSelectionList::append(std::string theNamingName) +void Model_AttributeSelectionList::append( + const std::string theNamingName, const std::string& theType) { int aNewTag = mySize->Get() + 1; TDF_Label aNewLab = mySize->Label().FindChild(aNewTag); @@ -55,8 +63,9 @@ void Model_AttributeSelectionList::append(std::string theNamingName) if (owner()) { aNewAttr->setObject(owner()); } + aNewAttr->setID(id()); mySize->Set(aNewTag); - aNewAttr->selectSubShape(selectionType(), theNamingName); + aNewAttr->selectSubShape(theType.empty() ? selectionType() : theType, theNamingName); owner()->data()->sendAttributeUpdated(this); } @@ -71,6 +80,58 @@ void Model_AttributeSelectionList::removeLast() aOldAttr->setObject(owner()); REMOVE_BACK_REF(aOldAttr->context()); aLab.ForgetAllAttributes(Standard_True); + myTmpAttr.reset(); + owner()->data()->sendAttributeUpdated(this); + } +} + +/// makes copy of all attributes on the given label and all sub-labels +static void copyAttrs(TDF_Label theSource, TDF_Label theDestination) { + TDF_AttributeIterator anAttrIter(theSource); + for(; anAttrIter.More(); anAttrIter.Next()) { + Handle(TDF_Attribute) aTargetAttr; + if (!theDestination.FindAttribute(anAttrIter.Value()->ID(), aTargetAttr)) { + // create a new attribute if not yet exists in the destination + aTargetAttr = anAttrIter.Value()->NewEmpty(); + theDestination.AddAttribute(aTargetAttr); + } + Handle(TDF_RelocationTable) aRelocTable = new TDF_RelocationTable(); // no relocation, empty map + anAttrIter.Value()->Paste(aTargetAttr, aRelocTable); + } + // copy the sub-labels content + TDF_ChildIterator aSubLabsIter(theSource); + for(; aSubLabsIter.More(); aSubLabsIter.Next()) { + copyAttrs(aSubLabsIter.Value(), theDestination.FindChild(aSubLabsIter.Value().Tag())); + } +} + +void Model_AttributeSelectionList::remove(const std::set& theIndices) +{ + int anOldSize = mySize->Get(); + int aRemoved = 0; + // iterate one by one and shifting the removed indicies + for(int aCurrent = 0; aCurrent < anOldSize; aCurrent++) { + if (theIndices.find(aCurrent) == theIndices.end()) { // not removed + if (aRemoved) { // but must be shifted to the removed position + TDF_Label aSource = mySize->Label().FindChild(aCurrent + 1); + TDF_Label aDest = mySize->Label().FindChild(aCurrent + 1 - aRemoved); + copyAttrs(aSource, aDest); + // remove the moved source + aSource.ForgetAllAttributes(Standard_True); + } + } else { // this is removed, so remove everything from this label + TDF_Label aLab = mySize->Label().FindChild(aCurrent + 1); + std::shared_ptr aOldAttr = + std::shared_ptr(new Model_AttributeSelection(aLab)); + aOldAttr->setObject(owner()); + REMOVE_BACK_REF(aOldAttr->context()); + aLab.ForgetAllAttributes(Standard_True); + myTmpAttr.reset(); + aRemoved++; + } + } + if (aRemoved) { // remove was performed, so, update the size and this attribute + mySize->Set(anOldSize - aRemoved); owner()->data()->sendAttributeUpdated(this); } } @@ -93,6 +154,9 @@ void Model_AttributeSelectionList::setSelectionType(const std::string& theType) std::shared_ptr Model_AttributeSelectionList::value(const int theIndex) { + if (myTmpAttr.get() && theIndex == size() - 1) { + return myTmpAttr; + } TDF_Label aLabel = mySize->Label().FindChild(theIndex + 1); // create a new attribute each time, by demand // supporting of old attributes is too slow (synch each time) and buggy on redo @@ -109,6 +173,7 @@ void Model_AttributeSelectionList::clear() { if (mySize->Get() != 0) { mySize->Set(0); + myTmpAttr.reset(); TDF_ChildIterator aSubIter(mySize->Label()); for(; aSubIter.More(); aSubIter.Next()) { TDF_Label aLab = aSubIter.Value();