1 // Copyright (C) 2014-2017 CEA/DEN, EDF R&D
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.
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.
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
17 // See http://www.salome-platform.org/ or
18 // email : webmaster.salome@opencascade.com<mailto:webmaster.salome@opencascade.com>
21 #include "Model_AttributeRefList.h"
22 #include "Model_Application.h"
23 #include "Model_Data.h"
24 #include "Model_Objects.h"
25 #include <ModelAPI_Feature.h>
26 #include <TDF_ListIteratorOfLabelList.hxx>
27 #include <TDF_Tool.hxx>
28 #include <TDataStd_ListIteratorOfListOfExtendedString.hxx>
30 void Model_AttributeRefList::append(ObjectPtr theObject)
32 if (owner()->document() == theObject->document()) {
33 std::shared_ptr<Model_Data> aData = std::dynamic_pointer_cast<Model_Data>(theObject->data());
34 myRef->Append(aData->label().Father()); // store label of the object
35 } else if (theObject.get() && theObject->data()->isValid()) { // reference to the other document
36 myRef->Append(myRef->Label());
37 // if these attributes exist, the link is external: keep reference to access the label
38 std::ostringstream anIdString; // string with document Id
39 anIdString<<theObject->document()->id();
40 myExtDocRef->Append(anIdString.str().c_str());
41 std::shared_ptr<Model_Data> aData = std::dynamic_pointer_cast<Model_Data>(theObject->data());
42 TCollection_AsciiString anEntry;
43 TDF_Tool::Entry(aData->label().Father(), anEntry);
44 myExtDocRef->Append(anEntry);
45 } else return; // something is wrong
48 myHashObjects.insert(theObject);
49 myHashIndex[myRef->Extent() - 1] = theObject;
50 myHashIndexNoEmpty[int(myHashIndexNoEmpty.size())] = theObject;
53 // do it before the transaction finish to make just created/removed objects know dependencies
54 // and reference from composite feature is removed automatically
55 ADD_BACK_REF(theObject);
56 owner()->data()->sendAttributeUpdated(this);
59 void Model_AttributeRefList::remove(ObjectPtr theObject)
62 if (theObject.get() != NULL) {
63 if (owner()->document() == theObject->document()) {
64 std::shared_ptr<Model_Data> aData;
65 aData = std::dynamic_pointer_cast<Model_Data>(theObject->data());
66 myRef->Remove(aData->label().Father());
67 REMOVE_BACK_REF(theObject);
68 owner()->data()->sendAttributeUpdated(this);
70 // create new lists because for the current moment remove one of the duplicated elements
71 // from the list is buggy
72 TDF_LabelList anOldList = myRef->List();
74 TDataStd_ListOfExtendedString anOldExts = myExtDocRef->List();
77 std::ostringstream anIdString; // string with document Id
78 anIdString<<theObject->document()->id();
79 std::shared_ptr<Model_Data> aData = std::dynamic_pointer_cast<Model_Data>(theObject->data());
80 TCollection_AsciiString anEntry;
81 TDF_Tool::Entry(aData->label().Father(), anEntry);
83 TDataStd_ListIteratorOfListOfExtendedString anExtIter(anOldExts);
84 for (TDF_ListIteratorOfLabelList aLIter(anOldList); aLIter.More(); aLIter.Next()) {
85 if (aLIter.Value() == myRef->Label()) {
86 if (anExtIter.Value() == anIdString.str().c_str()) {
87 TDataStd_ListIteratorOfListOfExtendedString anExtIter2 = anExtIter;
89 if (anExtIter2.Value() == anEntry) { // fully matches, so, remove(don't copy)
94 myExtDocRef->Append(anExtIter.Value());
96 myExtDocRef->Append(anExtIter.Value());
99 myRef->Append(aLIter.Value());
102 REMOVE_BACK_REF(theObject);
103 owner()->data()->sendAttributeUpdated(this);
107 else { // in case of empty object remove, the first empty object is removed from the list
108 std::shared_ptr<Model_Document> aDoc = std::dynamic_pointer_cast<Model_Document>(
109 owner()->document());
111 const TDF_LabelList& aList = myRef->List();
112 for (TDF_ListIteratorOfLabelList aLIter(aList); aLIter.More(); aLIter.Next()) {
113 ObjectPtr anObj = aDoc->objects()->object(aLIter.Value());
114 if (anObj.get() == NULL) {
115 myRef->Remove(aLIter.Value());
116 REMOVE_BACK_REF(theObject);
117 owner()->data()->sendAttributeUpdated(this);
125 void Model_AttributeRefList::clear()
127 std::list<ObjectPtr> anOldList = list();
129 std::list<ObjectPtr>::iterator anOldIter = anOldList.begin();
130 for(; anOldIter != anOldList.end(); anOldIter++) {
131 REMOVE_BACK_REF((*anOldIter));
133 myExtDocRef->Clear();
135 owner()->data()->sendAttributeUpdated(this);
138 int Model_AttributeRefList::size(const bool theWithEmpty) const
141 return myRef->Extent();
144 return int(myHashIndexNoEmpty.size());
147 const TDF_LabelList& aList = myRef->List();
148 for (TDF_ListIteratorOfLabelList aLIter(aList); aLIter.More(); aLIter.Next()) {
149 if (!aLIter.Value().IsNull() && !aLIter.Value().IsRoot()) aResult++;
154 bool Model_AttributeRefList::isInitialized()
156 if (size(false) == 0) {
157 // empty list is not initialized list: sketch will be not valid after add/undo
160 return ModelAPI_AttributeRefList::isInitialized();
163 ObjectPtr Model_AttributeRefList::iteratedObject(TDF_ListIteratorOfLabelList& theLIter,
164 TDataStd_ListIteratorOfListOfExtendedString& theExtIter,
165 std::shared_ptr<Model_Document> theDoc) const
168 if (!theLIter.Value().IsNull() && !theLIter.Value().IsRoot()) {
169 if (theLIter.Value() == myRef->Label()) { // external document object
170 int anID = atoi(TCollection_AsciiString(theExtIter.Value()).ToCString());
172 DocumentPtr aRefDoc = Model_Application::getApplication()->document(anID);
174 std::shared_ptr<Model_Document> aDR = std::dynamic_pointer_cast<Model_Document>(aRefDoc);
176 TDF_Tool::Label(aDR->objects()->featuresLabel().Data(),
177 TCollection_AsciiString(theExtIter.Value()).ToCString(), aRefLab);
178 if (!aRefLab.IsNull()) {
179 anObj = aDR->objects()->object(aRefLab);
183 } else { // internal document object
184 anObj = theDoc->objects()->object(theLIter.Value());
190 std::list<ObjectPtr> Model_AttributeRefList::list()
193 std::list<ObjectPtr> aResult;
194 std::map<int, ObjectPtr>::iterator aHashIter = myHashIndex.begin();
195 for(; aHashIter != myHashIndex.end(); aHashIter++) {
196 aResult.push_back(aHashIter->second);
201 bool Model_AttributeRefList::isInList(const ObjectPtr& theObj)
207 return myHashObjects.count(theObj) != 0;
210 ObjectPtr Model_AttributeRefList::object(const int theIndex, const bool theWithEmpty)
213 std::map<int, ObjectPtr>::iterator aFind;
215 aFind = myHashIndex.find(theIndex);
216 if (aFind == myHashIndex.end())
219 aFind = myHashIndexNoEmpty.find(theIndex);
220 if (aFind == myHashIndexNoEmpty.end())
223 return aFind->second;
226 void Model_AttributeRefList::substitute(const ObjectPtr& theCurrent, const ObjectPtr& theNew)
228 std::shared_ptr<Model_Document> aDoc = std::dynamic_pointer_cast<Model_Document>(
229 owner()->document());
231 std::shared_ptr<Model_Data> aData = std::dynamic_pointer_cast<Model_Data>(theCurrent->data());
232 if (aData.get() && aData->isValid()) {
233 TDF_Label aCurrentLab = aData->label().Father();
235 if (theNew.get() && theNew->data()->isValid()) { // the new may be null
236 std::shared_ptr<Model_Data> aNewData =
237 std::dynamic_pointer_cast<Model_Data>(theNew->data());
238 aNewLab = aNewData->label().Father();
240 aNewLab = aCurrentLab.Root(); // root means null object
242 // do the substitution
244 ADD_BACK_REF(theNew);
245 if (myRef->InsertAfter(aNewLab, aCurrentLab)) {
246 myRef->Remove(aCurrentLab);
247 REMOVE_BACK_REF(theCurrent);
249 owner()->data()->sendAttributeUpdated(this);
254 void Model_AttributeRefList::removeLast()
256 std::shared_ptr<Model_Document> aDoc = std::dynamic_pointer_cast<Model_Document>(
257 owner()->document());
258 if (aDoc && !myRef->IsEmpty()) {
259 ObjectPtr anObj = aDoc->objects()->object(myRef->Last());
262 myRef->Remove(myRef->Last());
263 REMOVE_BACK_REF(anObj);
264 owner()->data()->sendAttributeUpdated(this);
269 void Model_AttributeRefList::remove(const std::set<int>& theIndices)
271 std::shared_ptr<Model_Document> aDoc = std::dynamic_pointer_cast<Model_Document>(
272 owner()->document());
273 if (aDoc && !myRef->IsEmpty()) {
274 // collect labels that will be removed
275 TDF_LabelList aLabelsToRemove;
276 TDF_ListIteratorOfLabelList aLabIter(myRef->List());
277 for(int aCurrent = 0; aLabIter.More(); aLabIter.Next(), aCurrent++) {
278 if (theIndices.find(aCurrent) != theIndices.end())
279 aLabelsToRemove.Append(aLabIter.Value());
282 for(aLabIter.Initialize(aLabelsToRemove); aLabIter.More(); aLabIter.Next()) {
283 ObjectPtr anObj = aDoc->objects()->object(aLabIter.Value());
285 myRef->Remove(aLabIter.Value());
286 REMOVE_BACK_REF(anObj);
289 if (!aLabelsToRemove.IsEmpty()) {
291 owner()->data()->sendAttributeUpdated(this);
296 Model_AttributeRefList::Model_AttributeRefList(TDF_Label& theLabel)
302 void Model_AttributeRefList::reinit()
304 myIsInitialized = myLab.FindAttribute(TDataStd_ReferenceList::GetID(), myRef) == Standard_True;
305 if (!myIsInitialized) {
306 myRef = TDataStd_ReferenceList::Set(myLab);
308 if (!myLab.FindAttribute(TDataStd_ExtStringList::GetID(), myExtDocRef)) {
309 myExtDocRef = TDataStd_ExtStringList::Set(myLab);
314 void Model_AttributeRefList::createHash()
319 std::shared_ptr<Model_Document> aDoc = std::dynamic_pointer_cast<Model_Document>(
320 owner()->document());
322 const TDF_LabelList& aList = myRef->List();
323 TDataStd_ListIteratorOfListOfExtendedString anExtIter(myExtDocRef->List());
324 for (TDF_ListIteratorOfLabelList aLIter(aList); aLIter.More(); aLIter.Next()) {
325 ObjectPtr anObj = iteratedObject(aLIter, anExtIter, aDoc);
326 myHashIndex[int(myHashIndex.size())] = anObj;
328 myHashIndexNoEmpty[int(myHashIndexNoEmpty.size())] = anObj;
329 myHashObjects.insert(anObj);
332 if (!myHashObjects.empty()) // on open document with multi-rotation referenced have no results
338 void Model_AttributeRefList::eraseHash()
340 myHashObjects.clear();
342 myHashIndexNoEmpty.clear();