From 024d6f77757824ce0bd9780c105e7d80137e359f Mon Sep 17 00:00:00 2001 From: snn Date: Fri, 10 Feb 2017 15:48:36 +0300 Subject: [PATCH] 0028985: XCAF data structures for generic text annotations linked to GD&T annotations and saved views 1. A tool to annotate items in the hierarchical product structure was added. The tool is located under fixed label 0:1:9. It operates two basic entities: notes and annotations located under 0:1:9:1 and 0:1:9:2 hives correspondingly. A note is an attribute derived from base class XCAFDoc_Note that is attached to a separate label under the notes hive. An annotated item is represented by XCAFDoc_AssemblyItemRef attribute attached to a separate label under the annotated items hive. Notes are linked to annotated items by means of XCAFDoc_GraphNode attribute, where notes play parent roles and annotated items - child roles. 2. XCAFDoc_AssemblyItemRef defines a weak reference to a label with optional attribute GUID or sub-shape index. 3. A capability to store note/annotation labels in XCAFDoc_ViewTool was added. 4. XDE User guide was updated --- dox/user_guides/xde/images/xde_notes001.png | Bin 0 -> 79235 bytes dox/user_guides/xde/xde.md | 150 ++ src/BinMXCAFDoc/BinMXCAFDoc.cxx | 26 +- .../BinMXCAFDoc_AssemblyItemRefDriver.cxx | 111 ++ .../BinMXCAFDoc_AssemblyItemRefDriver.hxx | 54 + .../BinMXCAFDoc_NoteBalloonDriver.cxx | 52 + .../BinMXCAFDoc_NoteBalloonDriver.hxx | 42 + .../BinMXCAFDoc_NoteBinDataDriver.cxx | 96 ++ .../BinMXCAFDoc_NoteBinDataDriver.hxx | 44 + .../BinMXCAFDoc_NoteCommentDriver.cxx | 91 ++ .../BinMXCAFDoc_NoteCommentDriver.hxx | 49 + src/BinMXCAFDoc/BinMXCAFDoc_NoteDriver.cxx | 68 + src/BinMXCAFDoc/BinMXCAFDoc_NoteDriver.hxx | 55 + .../BinMXCAFDoc_NotesToolDriver.cxx | 62 + .../BinMXCAFDoc_NotesToolDriver.hxx | 54 + src/BinMXCAFDoc/FILES | 12 + src/XCAFDoc/FILES | 15 + src/XCAFDoc/XCAFDoc.cxx | 33 + src/XCAFDoc/XCAFDoc.hxx | 9 +- src/XCAFDoc/XCAFDoc_AssemblyItemId.cxx | 123 ++ src/XCAFDoc/XCAFDoc_AssemblyItemId.hxx | 101 ++ src/XCAFDoc/XCAFDoc_AssemblyItemRef.cxx | 295 ++++ src/XCAFDoc/XCAFDoc_AssemblyItemRef.hxx | 154 ++ src/XCAFDoc/XCAFDoc_DocumentTool.cxx | 24 + src/XCAFDoc/XCAFDoc_DocumentTool.hxx | 11 +- src/XCAFDoc/XCAFDoc_Note.cxx | 103 ++ src/XCAFDoc/XCAFDoc_Note.hxx | 82 + src/XCAFDoc/XCAFDoc_NoteBalloon.cxx | 68 + src/XCAFDoc/XCAFDoc_NoteBalloon.hxx | 58 + src/XCAFDoc/XCAFDoc_NoteBinData.cxx | 195 +++ src/XCAFDoc/XCAFDoc_NoteBinData.hxx | 129 ++ src/XCAFDoc/XCAFDoc_NoteComment.cxx | 113 ++ src/XCAFDoc/XCAFDoc_NoteComment.hxx | 72 + src/XCAFDoc/XCAFDoc_NotesTool.cxx | 848 ++++++++++ src/XCAFDoc/XCAFDoc_NotesTool.hxx | 519 ++++++ src/XCAFDoc/XCAFDoc_PartId.hxx | 22 + src/XCAFDoc/XCAFDoc_ViewTool.cxx | 216 +++ src/XCAFDoc/XCAFDoc_ViewTool.hxx | 26 +- src/XDEDRAW/FILES | 2 + src/XDEDRAW/XDEDRAW.cxx | 2 + src/XDEDRAW/XDEDRAW_Notes.cxx | 1438 +++++++++++++++++ src/XDEDRAW/XDEDRAW_Notes.hxx | 35 + src/XmlMXCAFDoc/FILES | 12 + src/XmlMXCAFDoc/XmlMXCAFDoc.cxx | 20 +- src/XmlMXCAFDoc/XmlMXCAFDoc.hxx | 4 + .../XmlMXCAFDoc_AssemblyItemRefDriver.cxx | 117 ++ .../XmlMXCAFDoc_AssemblyItemRefDriver.hxx | 55 + .../XmlMXCAFDoc_NoteBalloonDriver.cxx | 52 + .../XmlMXCAFDoc_NoteBalloonDriver.hxx | 42 + .../XmlMXCAFDoc_NoteBinDataDriver.cxx | 121 ++ .../XmlMXCAFDoc_NoteBinDataDriver.hxx | 45 + .../XmlMXCAFDoc_NoteCommentDriver.cxx | 95 ++ .../XmlMXCAFDoc_NoteCommentDriver.hxx | 50 + src/XmlMXCAFDoc/XmlMXCAFDoc_NoteDriver.cxx | 78 + src/XmlMXCAFDoc/XmlMXCAFDoc_NoteDriver.hxx | 56 + .../XmlMXCAFDoc_NotesToolDriver.cxx | 62 + .../XmlMXCAFDoc_NotesToolDriver.hxx | 55 + tests/gdt/grids.list | 1 + tests/gdt/notes/A1 | 5 + tests/gdt/notes/A2 | 13 + tests/gdt/notes/A3 | 9 + tests/gdt/notes/A4 | 16 + tests/gdt/notes/A5 | 16 + tests/gdt/notes/B1 | 18 + tests/gdt/notes/B2 | 22 + tests/gdt/notes/B3 | 25 + tests/gdt/notes/B4 | 22 + tests/gdt/notes/B5 | 15 + tests/gdt/notes/B6 | 18 + tests/gdt/notes/B7 | 21 + tests/gdt/notes/C1 | 24 + tests/gdt/notes/C2 | 28 + tests/gdt/notes/C3 | 28 + tests/gdt/notes/C4 | 27 + tests/gdt/notes/C5 | 29 + tests/gdt/notes/C6 | 29 + tests/gdt/notes/C7 | 31 + tests/gdt/notes/C8 | 33 + tests/gdt/notes/C9 | 33 + tests/gdt/notes/D1 | 51 + tests/gdt/notes/begin | 21 + tests/gdt/notes/end | 20 + 82 files changed, 7061 insertions(+), 17 deletions(-) create mode 100644 dox/user_guides/xde/images/xde_notes001.png create mode 100644 src/BinMXCAFDoc/BinMXCAFDoc_AssemblyItemRefDriver.cxx create mode 100644 src/BinMXCAFDoc/BinMXCAFDoc_AssemblyItemRefDriver.hxx create mode 100644 src/BinMXCAFDoc/BinMXCAFDoc_NoteBalloonDriver.cxx create mode 100644 src/BinMXCAFDoc/BinMXCAFDoc_NoteBalloonDriver.hxx create mode 100644 src/BinMXCAFDoc/BinMXCAFDoc_NoteBinDataDriver.cxx create mode 100644 src/BinMXCAFDoc/BinMXCAFDoc_NoteBinDataDriver.hxx create mode 100644 src/BinMXCAFDoc/BinMXCAFDoc_NoteCommentDriver.cxx create mode 100644 src/BinMXCAFDoc/BinMXCAFDoc_NoteCommentDriver.hxx create mode 100644 src/BinMXCAFDoc/BinMXCAFDoc_NoteDriver.cxx create mode 100644 src/BinMXCAFDoc/BinMXCAFDoc_NoteDriver.hxx create mode 100644 src/BinMXCAFDoc/BinMXCAFDoc_NotesToolDriver.cxx create mode 100644 src/BinMXCAFDoc/BinMXCAFDoc_NotesToolDriver.hxx create mode 100644 src/XCAFDoc/XCAFDoc_AssemblyItemId.cxx create mode 100644 src/XCAFDoc/XCAFDoc_AssemblyItemId.hxx create mode 100644 src/XCAFDoc/XCAFDoc_AssemblyItemRef.cxx create mode 100644 src/XCAFDoc/XCAFDoc_AssemblyItemRef.hxx create mode 100644 src/XCAFDoc/XCAFDoc_Note.cxx create mode 100644 src/XCAFDoc/XCAFDoc_Note.hxx create mode 100644 src/XCAFDoc/XCAFDoc_NoteBalloon.cxx create mode 100644 src/XCAFDoc/XCAFDoc_NoteBalloon.hxx create mode 100644 src/XCAFDoc/XCAFDoc_NoteBinData.cxx create mode 100644 src/XCAFDoc/XCAFDoc_NoteBinData.hxx create mode 100644 src/XCAFDoc/XCAFDoc_NoteComment.cxx create mode 100644 src/XCAFDoc/XCAFDoc_NoteComment.hxx create mode 100644 src/XCAFDoc/XCAFDoc_NotesTool.cxx create mode 100644 src/XCAFDoc/XCAFDoc_NotesTool.hxx create mode 100644 src/XCAFDoc/XCAFDoc_PartId.hxx create mode 100644 src/XDEDRAW/XDEDRAW_Notes.cxx create mode 100644 src/XDEDRAW/XDEDRAW_Notes.hxx create mode 100644 src/XmlMXCAFDoc/XmlMXCAFDoc_AssemblyItemRefDriver.cxx create mode 100644 src/XmlMXCAFDoc/XmlMXCAFDoc_AssemblyItemRefDriver.hxx create mode 100644 src/XmlMXCAFDoc/XmlMXCAFDoc_NoteBalloonDriver.cxx create mode 100644 src/XmlMXCAFDoc/XmlMXCAFDoc_NoteBalloonDriver.hxx create mode 100644 src/XmlMXCAFDoc/XmlMXCAFDoc_NoteBinDataDriver.cxx create mode 100644 src/XmlMXCAFDoc/XmlMXCAFDoc_NoteBinDataDriver.hxx create mode 100644 src/XmlMXCAFDoc/XmlMXCAFDoc_NoteCommentDriver.cxx create mode 100644 src/XmlMXCAFDoc/XmlMXCAFDoc_NoteCommentDriver.hxx create mode 100644 src/XmlMXCAFDoc/XmlMXCAFDoc_NoteDriver.cxx create mode 100644 src/XmlMXCAFDoc/XmlMXCAFDoc_NoteDriver.hxx create mode 100644 src/XmlMXCAFDoc/XmlMXCAFDoc_NotesToolDriver.cxx create mode 100644 src/XmlMXCAFDoc/XmlMXCAFDoc_NotesToolDriver.hxx create mode 100644 tests/gdt/notes/A1 create mode 100644 tests/gdt/notes/A2 create mode 100644 tests/gdt/notes/A3 create mode 100644 tests/gdt/notes/A4 create mode 100644 tests/gdt/notes/A5 create mode 100644 tests/gdt/notes/B1 create mode 100644 tests/gdt/notes/B2 create mode 100644 tests/gdt/notes/B3 create mode 100644 tests/gdt/notes/B4 create mode 100644 tests/gdt/notes/B5 create mode 100644 tests/gdt/notes/B6 create mode 100644 tests/gdt/notes/B7 create mode 100644 tests/gdt/notes/C1 create mode 100644 tests/gdt/notes/C2 create mode 100644 tests/gdt/notes/C3 create mode 100644 tests/gdt/notes/C4 create mode 100644 tests/gdt/notes/C5 create mode 100644 tests/gdt/notes/C6 create mode 100644 tests/gdt/notes/C7 create mode 100644 tests/gdt/notes/C8 create mode 100644 tests/gdt/notes/C9 create mode 100644 tests/gdt/notes/D1 create mode 100644 tests/gdt/notes/begin create mode 100644 tests/gdt/notes/end diff --git a/dox/user_guides/xde/images/xde_notes001.png b/dox/user_guides/xde/images/xde_notes001.png new file mode 100644 index 0000000000000000000000000000000000000000..2d1feae624712fbb58cef9333a34b958560ee46a GIT binary patch literal 79235 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%0Ahn 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

SGveY24i)xx9Bh^3t5pq(bq~xhQ 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 IrOiYB51qoi-~a#s literal 0 HcmV?d00001 diff --git a/dox/user_guides/xde/xde.md b/dox/user_guides/xde/xde.md index d87c3489d9..59e806c5ce 100644 --- a/dox/user_guides/xde/xde.md +++ b/dox/user_guides/xde/xde.md @@ -91,6 +91,26 @@ XDE can read and write colors and layers assigned to shapes or their subparts (d @figure{/user_guides/xde/images/xde_image006.png,"Colors and Layers",240} +@subsection occt_xde_1_7 Custom notes + +Custom notes is a kind of application specific data attached to assembly items, their attributes and sub-shapes. Basically, there are simple textual comments, binary data and other application specific data. Each note is provided with a timestamp and the user created it. + +Notes API provides the following functionality: + * Returns total number of notes and annotated items + * Returns labels for all notes and annotated items + * Creates notes: + - Comment note from a text string + - Binary data note from a file or byte array + * Checks if an assembly item is annotated + * Finds a label for the annotated item + * Returns all note labels for the annotated item + * Add a note to item(s): + - Assembly item + - Assembly item attribute + - Assembly item subshape index + * Remove note(s) from an annotated assembly item; orphan note(s) might be deleted optionally (items without linked notes will be deleted automatically) + * Delete note(s) and removes them from annotated items + * Get / delete orphan notes @section occt_xde_2 Working with XDE @@ -607,6 +627,136 @@ To remove a Color and all the references to it (so that the related shapes will myColors->RemoveColor(ColLabel); ~~~~~ +@subsection occt_xde_2_7 Custom notes + +In an XDE document, custom notes are managed by the class *XCAFDoc_NotesTool*. This is done with the same principles as for ShapeTool with Shapes, and with the same capability of having a tool on the Main Label, or on any sub-label. The Property itself is defined as sub-classes of an *XCAFDoc_Note* abstract class, which is a sub-class of *TDF_Attribute* one. + +Custom notes are stored in a child of the *XCAFDoc_NotesTool* label: it is at label 0.1.9.1. Each note then corresponds to a dedicated label. A note may be attached to a document item identified by a label, a sub-shape identified by integer index or an attribute identified by GUID. Annotations are stored in a child of the *XCAFDoc_NotesTool* label: it is at label 0.1.9.2. +Notes binding is done through *XCAFDoc_GraphNode* attribute. + + @figure{/user_guides/xde/images/xde_notes001.png,"Structure of notes part of XCAF document",240} + +@subsubsection occt_xde_2_7_1 Initialization + +To query, edit, or initialize a Document to handle custom notes of XCAF, use: +~~~~~ +Handle(XCAFDoc_NotesTool) myNotes = +XCAFDoc_DocumentTool::NotesTool(Doc->Main ()); +~~~~~ +This call can be used at any time. The first time it is used, a relevant structure is added to the document. This definition is used for all the following notes calls and will not be repeated for these. + +@subsubsection occt_xde_2_7_2 Creating Notes + +Before annotating a Document item a note must be created using one of the following methods of *XCAFDoc_NotesTool* class: +- CreateComment : creates a note with a textual comment +- CreateBinData : creates a note with arbitrary binary data, e.g. contents of a file + +Both methods return an instance of *XCAFDoc_Note* class. +~~~~~ +Handle(XCAFDoc_NotesTool) myNotes = ... +Handle(XCAFDoc_Note) myNote = myNotes->CreateComment("User", "Timestamp", "Hello, World!"); +~~~~~ +This code adds a child label to label 0.1.9.1 with *XCAFDoc_NoteComment* attribute. + +@subsubsection occt_xde_2_7_3 Editing a Note +An instance of *XCAFDoc_Note* class can be used for note editing. +One may change common note data. +~~~~~ +myNote->Set("New User", "New Timestamp"); +~~~~~ +To change specific data one need to down cast *myNote* handle to the appropriate sub-class: +~~~~~ +Handle(XCAFDoc_NoteComment) myCommentNote = Handle(XCAFDoc_NoteComment)::DownCast(myNote); +if (!myCommentNote.IsNull()) { + myCommentNote->Set("New comment"); +} +~~~~~ + +@subsubsection occt_xde_2_7_4 Adding Notes + +Once a note has been created it can be bound to a Document item using the following *XCAFDoc_NotesTool* methods: +- AddNote : binds a note to a label +- AddNoteToAttr : binds a note to a label's attribute +- AddNoteToSubshape : binds a note to a sub-shape + +All methods return a pointer to *XCAFDoc_AssemblyItemRef* attribute identifying the annotated item. +~~~~~ +Handle(XCAFDoc_NotesTool) myNotes = ... +Handle(XCAFDoc_Note) myNote = ... +TDF_Label theLabel; ... +Handle(XCAFDoc_AssemblyItemRef) myRef = myNotes->AddNote(myNote->Label(), theLabel); +Standard_GUID theAttrGUID; ... +Handle(XCAFDoc_AssemblyItemRef) myRefAttr = myNotes->AddNoteToAttr(myNote->Label(), theAttrGUID); +Standard_Integer theSubshape = 1; +Handle(XCAFDoc_AssemblyItemRef) myRefSubshape = myNotes->AddNoteToSubshape(myNote->Label(), theSubshape); +~~~~~ +This code adds three child labels to label 0.1.9.2 with *XCAFDoc_AssemblyItemRef* attribute with *XCAFDoc_GraphNode* attributes added to this and note labels. + +@subsubsection occt_xde_2_7_5 Finding Notes + +To find annotation labels under label 0.1.9.2 use the following *XCAFDoc_NotesTool* methods: +- FindAnnotatedItem : returns an annotation label for a label +- FindAnnotatedItemAttr : returns an annotation label for a label's attribute +- FindAnnotatedItemSubshape : returns an annotation label for a sub-shape + +~~~~~ +Handle(XCAFDoc_NotesTool) myNotes = ... +TDF_Label theLabel; ... +TDF_Label myLabel = myNotes->FindAnnotatedItem(theLabel); +Standard_GUID theAttrGUID; ... +TDF_Label myLabelAttr = myNotes->FindAnnotatedItemAttr(theLabel, theAttrGUID); +Standard_Integer theSubshape = 1; +TDF_Label myLabelSubshape = myNotes->FindAnnotatedItemSubshape(theLabel, theSubshape); +~~~~~ +Null label will be returned if there is no corresponding annotation. + +To get all notes of the Document item use the following *XCAFDoc_NotesTool* methods: +- GetNotes : outputs a sequence of note labels bound to a label +- GetAttrNotes : outputs a sequence of note labels bound to a label's attribute +- GetAttrSubshape : outputs a sequence of note labels bound to a sub-shape + +All these methods return the number of notes. +~~~~~ +Handle(XCAFDoc_NotesTool) myNotes = ... +TDF_Label theLabel; ... +TDF_LabelSequence theNotes; +myNotes->GetNotes(theLabel, theNotes); +Standard_GUID theAttrGUID; ... +TDF_LabelSequence theNotesAttr; +myNotes->GetAttrNotes(theLabel, theAttrGUID, theNotesAttr); +Standard_Integer theSubshape = 1; +TDF_LabelSequence theNotesSubshape; +myNotes->GetAttrSubshape(theLabel, theSubshape, theNotesSubshape); +~~~~~ + +@subsubsection occt_xde_2_7_6 Removing Notes + +To remove a note use one of the following *XCAFDoc_NotesTool* methods: +- RemoveNote : unbinds a note from a label +- RemoveAttrNote : unbinds a note from a label's attribute +- RemoveSubshapeNote : unbinds a note from a sub-shape + +~~~~~ +Handle(XCAFDoc_Note) myNote = ... +TDF_Label theLabel; ... +myNotes->RemoveNote(myNote->Label(), theLabel); +Standard_GUID theAttrGUID; ... +myRefAttr = myNotes->RemoveAttrNote(myNote->Label(), theAttrGUID); +Standard_Integer theSubshape = 1; +myNotes->RemoveSubshapeNote(myNote->Label(), theSubshape); +~~~~~ +A note will not be deleted automatically. +Counterpart methods to remove all notes are available too. + +@subsubsection occt_xde_2_7_7 Deleting Notes + +To delete note(s) use the following *XCAFDoc_NotesTool* methods: +- DeleteNote : deletes a single note +- DeleteNotes : deletes a sequence of notes +- DeleteAllNotes : deletes all Document notes +- DeleteOrphanNotes : deletes notes not bound to Document items + +All these methods excepting the last one break all links with Document items as well. @subsection occt_xde_2_8 Reading and Writing STEP or IGES Note that saving and restoring the document itself are standard OCAF operations. As the various previously described definitions enter into this frame, they will not be explained any further. diff --git a/src/BinMXCAFDoc/BinMXCAFDoc.cxx b/src/BinMXCAFDoc/BinMXCAFDoc.cxx index c0ba1d8d57..0ebe7bfedf 100644 --- a/src/BinMXCAFDoc/BinMXCAFDoc.cxx +++ b/src/BinMXCAFDoc/BinMXCAFDoc.cxx @@ -17,6 +17,7 @@ #include #include #include +#include #include #include #include @@ -32,6 +33,11 @@ #include #include #include +#include +#include +#include +#include +#include #include #include #include @@ -44,7 +50,8 @@ //purpose : //======================================================================= void BinMXCAFDoc::AddDrivers(const Handle(BinMDF_ADriverTable)& theDriverTable, - const Handle(CDM_MessageDriver)& theMsgDrv) { + const Handle(CDM_MessageDriver)& theMsgDrv) +{ theDriverTable->AddDriver( new BinMXCAFDoc_AreaDriver (theMsgDrv)); theDriverTable->AddDriver( new BinMXCAFDoc_CentroidDriver (theMsgDrv)); theDriverTable->AddDriver( new BinMXCAFDoc_ColorDriver (theMsgDrv)); @@ -63,13 +70,17 @@ void BinMXCAFDoc::AddDrivers(const Handle(BinMDF_ADriverTable)& theDriverTable, } theDriverTable->AddDriver( aLocationDriver); - theDriverTable->AddDriver( new BinMXCAFDoc_VolumeDriver (theMsgDrv)); - theDriverTable->AddDriver( new BinMXCAFDoc_DatumDriver (theMsgDrv)); + theDriverTable->AddDriver( new BinMXCAFDoc_AssemblyItemRefDriver(theMsgDrv)); + theDriverTable->AddDriver( new BinMXCAFDoc_VolumeDriver (theMsgDrv)); + theDriverTable->AddDriver( new BinMXCAFDoc_DatumDriver (theMsgDrv)); theDriverTable->AddDriver( new BinMXCAFDoc_GeomToleranceDriver (theMsgDrv)); - theDriverTable->AddDriver( new BinMXCAFDoc_DimensionDriver (theMsgDrv)); - theDriverTable->AddDriver( new BinMXCAFDoc_DimTolDriver (theMsgDrv)); - theDriverTable->AddDriver( new BinMXCAFDoc_MaterialDriver (theMsgDrv)); - theDriverTable->AddDriver( new BinMXCAFDoc_ViewDriver (theMsgDrv)); + theDriverTable->AddDriver( new BinMXCAFDoc_DimensionDriver (theMsgDrv)); + theDriverTable->AddDriver( new BinMXCAFDoc_DimTolDriver (theMsgDrv)); + theDriverTable->AddDriver( new BinMXCAFDoc_MaterialDriver (theMsgDrv)); + theDriverTable->AddDriver( new BinMXCAFDoc_NoteBalloonDriver (theMsgDrv)); + theDriverTable->AddDriver( new BinMXCAFDoc_NoteBinDataDriver (theMsgDrv)); + theDriverTable->AddDriver( new BinMXCAFDoc_NoteCommentDriver (theMsgDrv)); + theDriverTable->AddDriver( new BinMXCAFDoc_ViewDriver (theMsgDrv)); theDriverTable->AddDriver( new BinMXCAFDoc_ColorToolDriver (theMsgDrv)); theDriverTable->AddDriver( new BinMXCAFDoc_DocumentToolDriver(theMsgDrv)); @@ -77,5 +88,6 @@ void BinMXCAFDoc::AddDrivers(const Handle(BinMDF_ADriverTable)& theDriverTable, theDriverTable->AddDriver( new BinMXCAFDoc_ShapeToolDriver (theMsgDrv)); theDriverTable->AddDriver( new BinMXCAFDoc_DimTolToolDriver (theMsgDrv)); theDriverTable->AddDriver( new BinMXCAFDoc_MaterialToolDriver(theMsgDrv)); + theDriverTable->AddDriver( new BinMXCAFDoc_NotesToolDriver (theMsgDrv)); theDriverTable->AddDriver( new BinMXCAFDoc_ViewToolDriver (theMsgDrv)); } diff --git a/src/BinMXCAFDoc/BinMXCAFDoc_AssemblyItemRefDriver.cxx b/src/BinMXCAFDoc/BinMXCAFDoc_AssemblyItemRefDriver.cxx new file mode 100644 index 0000000000..7d362ec27c --- /dev/null +++ b/src/BinMXCAFDoc/BinMXCAFDoc_AssemblyItemRefDriver.cxx @@ -0,0 +1,111 @@ +// Created on: 2017-02-16 +// Created by: Eugeny NIKONOV +// Copyright (c) 2005-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include +#include +#include +#include + +IMPLEMENT_STANDARD_RTTIEXT(BinMXCAFDoc_AssemblyItemRefDriver, BinMDF_ADriver) + +//======================================================================= +//function : +//purpose : +//======================================================================= +BinMXCAFDoc_AssemblyItemRefDriver::BinMXCAFDoc_AssemblyItemRefDriver(const Handle(CDM_MessageDriver)& theMsgDriver) + : BinMDF_ADriver(theMsgDriver, STANDARD_TYPE(XCAFDoc_AssemblyItemRef)->Name()) +{ + +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Handle(TDF_Attribute) BinMXCAFDoc_AssemblyItemRefDriver::NewEmpty() const +{ + return new XCAFDoc_AssemblyItemRef(); +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Standard_Boolean BinMXCAFDoc_AssemblyItemRefDriver::Paste(const BinObjMgt_Persistent& theSource, + const Handle(TDF_Attribute)& theTarget, + BinObjMgt_RRelocationTable& /*theRelocTable*/) const +{ + Handle(XCAFDoc_AssemblyItemRef) aThis = Handle(XCAFDoc_AssemblyItemRef)::DownCast(theTarget); + if (aThis.IsNull()) + return Standard_False; + + TCollection_AsciiString aPathStr; + if (!(theSource >> aPathStr)) + return Standard_False; + + aThis->SetItem(aPathStr); + + Standard_Integer anExtraRef = 0; + if (!(theSource >> anExtraRef)) + return Standard_False; + + if (anExtraRef == 1) + { + Standard_GUID aGUID; + if (!(theSource >> aGUID)) + return Standard_False; + + aThis->SetGUID(aGUID); + } + else if (anExtraRef == 2) + { + Standard_Integer aSubshapeIndex; + if (!(theSource >> aSubshapeIndex)) + return Standard_False; + + aThis->SetSubshapeIndex(aSubshapeIndex); + } + + return Standard_True; +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +void BinMXCAFDoc_AssemblyItemRefDriver::Paste(const Handle(TDF_Attribute)& theSource, + BinObjMgt_Persistent& theTarget, + BinObjMgt_SRelocationTable& /*theRelocTable*/) const +{ + Handle(XCAFDoc_AssemblyItemRef) aThis = Handle(XCAFDoc_AssemblyItemRef)::DownCast(theSource); + if (!aThis.IsNull()) + { + theTarget << aThis->GetItem().ToString(); + if (aThis->IsGUID()) + { + theTarget << Standard_Integer(1); + theTarget << aThis->GetGUID(); + } + else if (aThis->IsSubshapeIndex()) + { + theTarget << Standard_Integer(2); + theTarget << aThis->GetSubshapeIndex(); + } + else + theTarget << Standard_Integer(0); + } +} diff --git a/src/BinMXCAFDoc/BinMXCAFDoc_AssemblyItemRefDriver.hxx b/src/BinMXCAFDoc/BinMXCAFDoc_AssemblyItemRefDriver.hxx new file mode 100644 index 0000000000..9f031eae6d --- /dev/null +++ b/src/BinMXCAFDoc/BinMXCAFDoc_AssemblyItemRefDriver.hxx @@ -0,0 +1,54 @@ +// Created on: 2017-02-16 +// Created by: Sergey NIKONOV +// Copyright (c) 2005-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _BinMXCAFDoc_AssemblyItemRefDriver_HeaderFile +#define _BinMXCAFDoc_AssemblyItemRefDriver_HeaderFile + +#include +#include + +#include +#include +#include +#include + +class CDM_MessageDriver; +class TDF_Attribute; +class BinObjMgt_Persistent; + +class BinMXCAFDoc_AssemblyItemRefDriver; +DEFINE_STANDARD_HANDLE(BinMXCAFDoc_AssemblyItemRefDriver, BinMDF_ADriver) + +class BinMXCAFDoc_AssemblyItemRefDriver : public BinMDF_ADriver +{ +public: + + Standard_EXPORT BinMXCAFDoc_AssemblyItemRefDriver(const Handle(CDM_MessageDriver)& theMsgDriver); + + Standard_EXPORT Handle(TDF_Attribute) NewEmpty() const Standard_OVERRIDE; + + Standard_EXPORT Standard_Boolean Paste (const BinObjMgt_Persistent& theSource, + const Handle(TDF_Attribute)& theTarget, + BinObjMgt_RRelocationTable& theRelocTable) const Standard_OVERRIDE; + + Standard_EXPORT void Paste (const Handle(TDF_Attribute)& theSource, + BinObjMgt_Persistent& theTarget, + BinObjMgt_SRelocationTable& theRelocTable) const Standard_OVERRIDE; + + DEFINE_STANDARD_RTTIEXT(BinMXCAFDoc_AssemblyItemRefDriver, BinMDF_ADriver) + +}; + +#endif // _BinMXCAFDoc_AssemblyItemRefDriver_HeaderFile diff --git a/src/BinMXCAFDoc/BinMXCAFDoc_NoteBalloonDriver.cxx b/src/BinMXCAFDoc/BinMXCAFDoc_NoteBalloonDriver.cxx new file mode 100644 index 0000000000..700ec100f7 --- /dev/null +++ b/src/BinMXCAFDoc/BinMXCAFDoc_NoteBalloonDriver.cxx @@ -0,0 +1,52 @@ +// Created on: 2017-08-10 +// Created by: Eugeny NIKONOV +// Copyright (c) 2005-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include +#include +#include +#include + +IMPLEMENT_STANDARD_RTTIEXT(BinMXCAFDoc_NoteBalloonDriver, BinMXCAFDoc_NoteCommentDriver) + +//======================================================================= +//function : +//purpose : +//======================================================================= +BinMXCAFDoc_NoteBalloonDriver::BinMXCAFDoc_NoteBalloonDriver(const Handle(CDM_MessageDriver)& theMsgDriver) + : BinMXCAFDoc_NoteCommentDriver(theMsgDriver, STANDARD_TYPE(XCAFDoc_NoteBalloon)->Name()) +{ +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Handle(TDF_Attribute) BinMXCAFDoc_NoteBalloonDriver::NewEmpty() const +{ + return new XCAFDoc_NoteBalloon(); +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +BinMXCAFDoc_NoteBalloonDriver::BinMXCAFDoc_NoteBalloonDriver(const Handle(CDM_MessageDriver)& theMsgDriver, + Standard_CString theName) + : BinMXCAFDoc_NoteCommentDriver(theMsgDriver, theName) +{ + +} diff --git a/src/BinMXCAFDoc/BinMXCAFDoc_NoteBalloonDriver.hxx b/src/BinMXCAFDoc/BinMXCAFDoc_NoteBalloonDriver.hxx new file mode 100644 index 0000000000..444b77a53d --- /dev/null +++ b/src/BinMXCAFDoc/BinMXCAFDoc_NoteBalloonDriver.hxx @@ -0,0 +1,42 @@ +// Created on: 2017-08-10 +// Created by: Sergey NIKONOV +// Copyright (c) 2005-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _BinMXCAFDoc_NoteBalloonDriver_HeaderFile +#define _BinMXCAFDoc_NoteBalloonDriver_HeaderFile + +#include + +class BinMXCAFDoc_NoteBalloonDriver; +DEFINE_STANDARD_HANDLE(BinMXCAFDoc_NoteBalloonDriver, BinMXCAFDoc_NoteCommentDriver) + +class BinMXCAFDoc_NoteBalloonDriver : public BinMXCAFDoc_NoteCommentDriver +{ +public: + + Standard_EXPORT BinMXCAFDoc_NoteBalloonDriver(const Handle(CDM_MessageDriver)& theMsgDriver); + + Standard_EXPORT Handle(TDF_Attribute) NewEmpty() const Standard_OVERRIDE; + + DEFINE_STANDARD_RTTIEXT(BinMXCAFDoc_NoteBalloonDriver, BinMXCAFDoc_NoteCommentDriver) + + +protected: + + BinMXCAFDoc_NoteBalloonDriver(const Handle(CDM_MessageDriver)& theMsgDriver, + Standard_CString theName); + +}; + +#endif // _BinMXCAFDoc_NoteCommentDriver_HeaderFile diff --git a/src/BinMXCAFDoc/BinMXCAFDoc_NoteBinDataDriver.cxx b/src/BinMXCAFDoc/BinMXCAFDoc_NoteBinDataDriver.cxx new file mode 100644 index 0000000000..ee73b46636 --- /dev/null +++ b/src/BinMXCAFDoc/BinMXCAFDoc_NoteBinDataDriver.cxx @@ -0,0 +1,96 @@ +// Created on: 2017-02-13 +// Created by: Eugeny NIKONOV +// Copyright (c) 2005-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +IMPLEMENT_STANDARD_RTTIEXT(BinMXCAFDoc_NoteBinDataDriver, BinMXCAFDoc_NoteDriver) + +//======================================================================= +//function : +//purpose : +//======================================================================= +BinMXCAFDoc_NoteBinDataDriver::BinMXCAFDoc_NoteBinDataDriver(const Handle(CDM_MessageDriver)& theMsgDriver) + : BinMXCAFDoc_NoteDriver(theMsgDriver, STANDARD_TYPE(XCAFDoc_NoteBinData)->Name()) +{ +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Handle(TDF_Attribute) BinMXCAFDoc_NoteBinDataDriver::NewEmpty() const +{ + return new XCAFDoc_NoteBinData(); +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Standard_Boolean BinMXCAFDoc_NoteBinDataDriver::Paste(const BinObjMgt_Persistent& theSource, + const Handle(TDF_Attribute)& theTarget, + BinObjMgt_RRelocationTable& theRelocTable) const +{ + if (!BinMXCAFDoc_NoteDriver::Paste(theSource, theTarget, theRelocTable)) + return Standard_False; + + Handle(XCAFDoc_NoteBinData) aNote = Handle(XCAFDoc_NoteBinData)::DownCast(theTarget); + if (aNote.IsNull()) + return Standard_False; + + TCollection_ExtendedString aTitle; + TCollection_AsciiString aMIMEtype; + Standard_Integer nbSize; + if (!(theSource >> aTitle >> aMIMEtype >> nbSize)) + return Standard_False; + + Handle(TColStd_HArray1OfByte) aData; + if (nbSize > 0) + { + aData.reset(new TColStd_HArray1OfByte(1, nbSize)); + theSource.GetByteArray(&aData->ChangeFirst(), nbSize); + } + + aNote->Set(aTitle, aMIMEtype, aData); + + return Standard_True; +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +void BinMXCAFDoc_NoteBinDataDriver::Paste(const Handle(TDF_Attribute)& theSource, + BinObjMgt_Persistent& theTarget, + BinObjMgt_SRelocationTable& theRelocTable) const +{ + BinMXCAFDoc_NoteDriver::Paste(theSource, theTarget, theRelocTable); + + Handle(XCAFDoc_NoteBinData) aNote = Handle(XCAFDoc_NoteBinData)::DownCast(theSource); + if (!aNote.IsNull()) + { + theTarget << aNote->Title() << aNote->MIMEtype() << aNote->Size(); + if (aNote->Size() > 0) + theTarget.PutByteArray(&aNote->Data()->ChangeFirst(), aNote->Size()); + } +} diff --git a/src/BinMXCAFDoc/BinMXCAFDoc_NoteBinDataDriver.hxx b/src/BinMXCAFDoc/BinMXCAFDoc_NoteBinDataDriver.hxx new file mode 100644 index 0000000000..422f79784d --- /dev/null +++ b/src/BinMXCAFDoc/BinMXCAFDoc_NoteBinDataDriver.hxx @@ -0,0 +1,44 @@ +// Created on: 2017-02-13 +// Created by: Sergey NIKONOV +// Copyright (c) 2005-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _BinMXCAFDoc_NoteBinDataDriver_HeaderFile +#define _BinMXCAFDoc_NoteBinDataDriver_HeaderFile + +#include + +class BinMXCAFDoc_NoteBinDataDriver; +DEFINE_STANDARD_HANDLE(BinMXCAFDoc_NoteBinDataDriver, BinMXCAFDoc_NoteDriver) + +class BinMXCAFDoc_NoteBinDataDriver : public BinMXCAFDoc_NoteDriver +{ +public: + + Standard_EXPORT BinMXCAFDoc_NoteBinDataDriver(const Handle(CDM_MessageDriver)& theMsgDriver); + + Standard_EXPORT Handle(TDF_Attribute) NewEmpty() const Standard_OVERRIDE; + + Standard_EXPORT Standard_Boolean Paste (const BinObjMgt_Persistent& theSource, + const Handle(TDF_Attribute)& theTarget, + BinObjMgt_RRelocationTable& theRelocTable) const Standard_OVERRIDE; + + Standard_EXPORT void Paste (const Handle(TDF_Attribute)& theSource, + BinObjMgt_Persistent& theTarget, + BinObjMgt_SRelocationTable& theRelocTable) const Standard_OVERRIDE; + + DEFINE_STANDARD_RTTIEXT(BinMXCAFDoc_NoteBinDataDriver, BinMXCAFDoc_NoteDriver) + +}; + +#endif // _BinMXCAFDoc_NoteBinDataDriver_HeaderFile diff --git a/src/BinMXCAFDoc/BinMXCAFDoc_NoteCommentDriver.cxx b/src/BinMXCAFDoc/BinMXCAFDoc_NoteCommentDriver.cxx new file mode 100644 index 0000000000..e550915508 --- /dev/null +++ b/src/BinMXCAFDoc/BinMXCAFDoc_NoteCommentDriver.cxx @@ -0,0 +1,91 @@ +// Created on: 2017-02-13 +// Created by: Eugeny NIKONOV +// Copyright (c) 2005-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include +#include +#include +#include + +IMPLEMENT_STANDARD_RTTIEXT(BinMXCAFDoc_NoteCommentDriver, BinMXCAFDoc_NoteDriver) + +//======================================================================= +//function : +//purpose : +//======================================================================= +BinMXCAFDoc_NoteCommentDriver::BinMXCAFDoc_NoteCommentDriver(const Handle(CDM_MessageDriver)& theMsgDriver) + : BinMXCAFDoc_NoteDriver(theMsgDriver, STANDARD_TYPE(XCAFDoc_NoteComment)->Name()) +{ +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Handle(TDF_Attribute) BinMXCAFDoc_NoteCommentDriver::NewEmpty() const +{ + return new XCAFDoc_NoteComment(); +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Standard_Boolean BinMXCAFDoc_NoteCommentDriver::Paste(const BinObjMgt_Persistent& theSource, + const Handle(TDF_Attribute)& theTarget, + BinObjMgt_RRelocationTable& theRelocTable) const +{ + if (!BinMXCAFDoc_NoteDriver::Paste(theSource, theTarget, theRelocTable)) + return Standard_False; + + Handle(XCAFDoc_NoteComment) aNote = Handle(XCAFDoc_NoteComment)::DownCast(theTarget); + if (aNote.IsNull()) + return Standard_False; + + TCollection_ExtendedString aComment; + if (!(theSource >> aComment)) + return Standard_False; + + aNote->Set(aComment); + + return Standard_True; +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +void BinMXCAFDoc_NoteCommentDriver::Paste(const Handle(TDF_Attribute)& theSource, + BinObjMgt_Persistent& theTarget, + BinObjMgt_SRelocationTable& theRelocTable) const +{ + BinMXCAFDoc_NoteDriver::Paste(theSource, theTarget, theRelocTable); + + Handle(XCAFDoc_NoteComment) aNote = Handle(XCAFDoc_NoteComment)::DownCast(theSource); + if (!aNote.IsNull()) + theTarget << aNote->Comment(); +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +BinMXCAFDoc_NoteCommentDriver::BinMXCAFDoc_NoteCommentDriver(const Handle(CDM_MessageDriver)& theMsgDriver, + Standard_CString theName) + : BinMXCAFDoc_NoteDriver(theMsgDriver, theName) +{ + +} diff --git a/src/BinMXCAFDoc/BinMXCAFDoc_NoteCommentDriver.hxx b/src/BinMXCAFDoc/BinMXCAFDoc_NoteCommentDriver.hxx new file mode 100644 index 0000000000..cad5b6b198 --- /dev/null +++ b/src/BinMXCAFDoc/BinMXCAFDoc_NoteCommentDriver.hxx @@ -0,0 +1,49 @@ +// Created on: 2017-02-13 +// Created by: Sergey NIKONOV +// Copyright (c) 2005-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _BinMXCAFDoc_NoteCommentDriver_HeaderFile +#define _BinMXCAFDoc_NoteCommentDriver_HeaderFile + +#include + +class BinMXCAFDoc_NoteCommentDriver; +DEFINE_STANDARD_HANDLE(BinMXCAFDoc_NoteCommentDriver, BinMXCAFDoc_NoteDriver) + +class BinMXCAFDoc_NoteCommentDriver : public BinMXCAFDoc_NoteDriver +{ +public: + + Standard_EXPORT BinMXCAFDoc_NoteCommentDriver(const Handle(CDM_MessageDriver)& theMsgDriver); + + Standard_EXPORT Handle(TDF_Attribute) NewEmpty() const Standard_OVERRIDE; + + Standard_EXPORT Standard_Boolean Paste (const BinObjMgt_Persistent& theSource, + const Handle(TDF_Attribute)& theTarget, + BinObjMgt_RRelocationTable& theRelocTable) const Standard_OVERRIDE; + + Standard_EXPORT void Paste (const Handle(TDF_Attribute)& theSource, + BinObjMgt_Persistent& theTarget, + BinObjMgt_SRelocationTable& theRelocTable) const Standard_OVERRIDE; + + DEFINE_STANDARD_RTTIEXT(BinMXCAFDoc_NoteCommentDriver, BinMXCAFDoc_NoteDriver) + +protected: + + BinMXCAFDoc_NoteCommentDriver(const Handle(CDM_MessageDriver)& theMsgDriver, + Standard_CString theName); + +}; + +#endif // _BinMXCAFDoc_NoteCommentDriver_HeaderFile diff --git a/src/BinMXCAFDoc/BinMXCAFDoc_NoteDriver.cxx b/src/BinMXCAFDoc/BinMXCAFDoc_NoteDriver.cxx new file mode 100644 index 0000000000..fc00d0b83c --- /dev/null +++ b/src/BinMXCAFDoc/BinMXCAFDoc_NoteDriver.cxx @@ -0,0 +1,68 @@ +// Created on: 2017-02-10 +// Created by: Eugeny NIKONOV +// Copyright (c) 2005-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include +#include +#include +#include + +IMPLEMENT_STANDARD_RTTIEXT(BinMXCAFDoc_NoteDriver, BinMDF_ADriver) + +//======================================================================= +//function : +//purpose : +//======================================================================= +BinMXCAFDoc_NoteDriver::BinMXCAFDoc_NoteDriver(const Handle(CDM_MessageDriver)& theMsgDriver, + Standard_CString theName) + : BinMDF_ADriver(theMsgDriver, theName) +{ + +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Standard_Boolean BinMXCAFDoc_NoteDriver::Paste(const BinObjMgt_Persistent& theSource, + const Handle(TDF_Attribute)& theTarget, + BinObjMgt_RRelocationTable& /*theRelocTable*/) const +{ + Handle(XCAFDoc_Note) aNote = Handle(XCAFDoc_Note)::DownCast(theTarget); + if (aNote.IsNull()) + return Standard_False; + + TCollection_ExtendedString aUserName, aTimeStamp; + if (!(theSource >> aUserName >> aTimeStamp)) + return Standard_False; + + aNote->Set(aUserName, aTimeStamp); + + return Standard_True; +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +void BinMXCAFDoc_NoteDriver::Paste(const Handle(TDF_Attribute)& theSource, + BinObjMgt_Persistent& theTarget, + BinObjMgt_SRelocationTable& /*theRelocTable*/) const +{ + Handle(XCAFDoc_Note) aNote = Handle(XCAFDoc_Note)::DownCast(theSource); + if (!aNote.IsNull()) + theTarget << aNote->UserName() << aNote->TimeStamp(); +} diff --git a/src/BinMXCAFDoc/BinMXCAFDoc_NoteDriver.hxx b/src/BinMXCAFDoc/BinMXCAFDoc_NoteDriver.hxx new file mode 100644 index 0000000000..17900a0b1a --- /dev/null +++ b/src/BinMXCAFDoc/BinMXCAFDoc_NoteDriver.hxx @@ -0,0 +1,55 @@ +// Created on: 2017-02-10 +// Created by: Sergey NIKONOV +// Copyright (c) 2005-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _BinMXCAFDoc_NoteDriver_HeaderFile +#define _BinMXCAFDoc_NoteDriver_HeaderFile + +#include +#include + +#include +#include +#include +#include + +class CDM_MessageDriver; +class TDF_Attribute; +class BinObjMgt_Persistent; + +class BinMXCAFDoc_NoteDriver; +DEFINE_STANDARD_HANDLE(BinMXCAFDoc_NoteDriver, BinMDF_ADriver) + +class BinMXCAFDoc_NoteDriver : public BinMDF_ADriver +{ +public: + + Standard_EXPORT Standard_Boolean Paste (const BinObjMgt_Persistent& theSource, + const Handle(TDF_Attribute)& theTarget, + BinObjMgt_RRelocationTable& theRelocTable) const Standard_OVERRIDE; + + Standard_EXPORT void Paste (const Handle(TDF_Attribute)& theSource, + BinObjMgt_Persistent& theTarget, + BinObjMgt_SRelocationTable& theRelocTable) const Standard_OVERRIDE; + + DEFINE_STANDARD_RTTIEXT(BinMXCAFDoc_NoteDriver, BinMDF_ADriver) + +protected: + + BinMXCAFDoc_NoteDriver(const Handle(CDM_MessageDriver)& theMsgDriver, + Standard_CString theName); + +}; + +#endif // _BinMXCAFDoc_NoteDriver_HeaderFile diff --git a/src/BinMXCAFDoc/BinMXCAFDoc_NotesToolDriver.cxx b/src/BinMXCAFDoc/BinMXCAFDoc_NotesToolDriver.cxx new file mode 100644 index 0000000000..a3979e2be5 --- /dev/null +++ b/src/BinMXCAFDoc/BinMXCAFDoc_NotesToolDriver.cxx @@ -0,0 +1,62 @@ +// Created on: 2017-02-10 +// Created by: Eugeny NIKONOV +// Copyright (c) 2005-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include +#include +#include +#include + +IMPLEMENT_STANDARD_RTTIEXT(BinMXCAFDoc_NotesToolDriver, BinMDF_ADriver) + +//======================================================================= +//function : +//purpose : +//======================================================================= +BinMXCAFDoc_NotesToolDriver::BinMXCAFDoc_NotesToolDriver(const Handle(CDM_MessageDriver)& theMsgDriver) + : BinMDF_ADriver(theMsgDriver, STANDARD_TYPE(XCAFDoc_NotesTool)->Name()) +{ +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Handle(TDF_Attribute) BinMXCAFDoc_NotesToolDriver::NewEmpty() const +{ + return new XCAFDoc_NotesTool(); +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Standard_Boolean BinMXCAFDoc_NotesToolDriver::Paste(const BinObjMgt_Persistent& /*theSource*/, + const Handle(TDF_Attribute)& /*theTarget*/, + BinObjMgt_RRelocationTable& /*theRelocTable*/) const +{ + return Standard_True; +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +void BinMXCAFDoc_NotesToolDriver::Paste(const Handle(TDF_Attribute)& /*theSource*/, + BinObjMgt_Persistent& /*theTarget*/, + BinObjMgt_SRelocationTable& /*theRelocTable*/) const +{ +} diff --git a/src/BinMXCAFDoc/BinMXCAFDoc_NotesToolDriver.hxx b/src/BinMXCAFDoc/BinMXCAFDoc_NotesToolDriver.hxx new file mode 100644 index 0000000000..3a8fa0a3b7 --- /dev/null +++ b/src/BinMXCAFDoc/BinMXCAFDoc_NotesToolDriver.hxx @@ -0,0 +1,54 @@ +// Created on: 2017-02-10 +// Created by: Sergey NIKONOV +// Copyright (c) 2005-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _BinMXCAFDoc_NotesToolDriver_HeaderFile +#define _BinMXCAFDoc_NotesToolDriver_HeaderFile + +#include +#include + +#include +#include +#include +#include + +class CDM_MessageDriver; +class TDF_Attribute; +class BinObjMgt_Persistent; + +class BinMXCAFDoc_NotesToolDriver; +DEFINE_STANDARD_HANDLE(BinMXCAFDoc_NotesToolDriver, BinMDF_ADriver) + +class BinMXCAFDoc_NotesToolDriver : public BinMDF_ADriver +{ +public: + + Standard_EXPORT BinMXCAFDoc_NotesToolDriver(const Handle(CDM_MessageDriver)& theMsgDriver); + + Standard_EXPORT Handle(TDF_Attribute) NewEmpty() const Standard_OVERRIDE; + + Standard_EXPORT Standard_Boolean Paste (const BinObjMgt_Persistent& theSource, + const Handle(TDF_Attribute)& theTarget, + BinObjMgt_RRelocationTable& theRelocTable) const Standard_OVERRIDE; + + Standard_EXPORT void Paste (const Handle(TDF_Attribute)& theSource, + BinObjMgt_Persistent& theTarget, + BinObjMgt_SRelocationTable& theRelocTable) const Standard_OVERRIDE; + + DEFINE_STANDARD_RTTIEXT(BinMXCAFDoc_NotesToolDriver, BinMDF_ADriver) + +}; + +#endif // _BinMXCAFDoc_NotesToolDriver_HeaderFile diff --git a/src/BinMXCAFDoc/FILES b/src/BinMXCAFDoc/FILES index fe29702e24..fbb3e87764 100644 --- a/src/BinMXCAFDoc/FILES +++ b/src/BinMXCAFDoc/FILES @@ -1,5 +1,7 @@ BinMXCAFDoc.cxx BinMXCAFDoc.hxx +BinMXCAFDoc_AssemblyItemRefDriver.cxx +BinMXCAFDoc_AssemblyItemRefDriver.hxx BinMXCAFDoc_AreaDriver.cxx BinMXCAFDoc_AreaDriver.hxx BinMXCAFDoc_CentroidDriver.cxx @@ -33,6 +35,16 @@ BinMXCAFDoc_MaterialDriver.cxx BinMXCAFDoc_MaterialDriver.hxx BinMXCAFDoc_MaterialToolDriver.cxx BinMXCAFDoc_MaterialToolDriver.hxx +BinMXCAFDoc_NoteDriver.cxx +BinMXCAFDoc_NoteDriver.hxx +BinMXCAFDoc_NoteBalloonDriver.cxx +BinMXCAFDoc_NoteBalloonDriver.hxx +BinMXCAFDoc_NoteCommentDriver.cxx +BinMXCAFDoc_NoteCommentDriver.hxx +BinMXCAFDoc_NoteBinDataDriver.cxx +BinMXCAFDoc_NoteBinDataDriver.hxx +BinMXCAFDoc_NotesToolDriver.cxx +BinMXCAFDoc_NotesToolDriver.hxx BinMXCAFDoc_ShapeToolDriver.cxx BinMXCAFDoc_ShapeToolDriver.hxx BinMXCAFDoc_ViewDriver.cxx diff --git a/src/XCAFDoc/FILES b/src/XCAFDoc/FILES index a30837c6fe..07384bb750 100755 --- a/src/XCAFDoc/FILES +++ b/src/XCAFDoc/FILES @@ -2,6 +2,11 @@ FILES GUID.txt XCAFDoc.cxx XCAFDoc.hxx +XCAFDoc_AssemblyItemId.cxx +XCAFDoc_AssemblyItemId.hxx +XCAFDoc_AssemblyItemRef.cxx +XCAFDoc_AssemblyItemRef.hxx +XCAFDoc_PartId.hxx XCAFDoc_Area.cxx XCAFDoc_Area.hxx XCAFDoc_Centroid.cxx @@ -40,6 +45,16 @@ XCAFDoc_Material.cxx XCAFDoc_Material.hxx XCAFDoc_MaterialTool.cxx XCAFDoc_MaterialTool.hxx +XCAFDoc_Note.cxx +XCAFDoc_Note.hxx +XCAFDoc_NoteBalloon.cxx +XCAFDoc_NoteBalloon.hxx +XCAFDoc_NoteComment.cxx +XCAFDoc_NoteComment.hxx +XCAFDoc_NoteBinData.cxx +XCAFDoc_NoteBinData.hxx +XCAFDoc_NotesTool.cxx +XCAFDoc_NotesTool.hxx XCAFDoc_ShapeMapTool.cxx XCAFDoc_ShapeMapTool.hxx XCAFDoc_ShapeTool.cxx diff --git a/src/XCAFDoc/XCAFDoc.cxx b/src/XCAFDoc/XCAFDoc.cxx index 2299ae3293..e92b36227e 100644 --- a/src/XCAFDoc/XCAFDoc.cxx +++ b/src/XCAFDoc/XCAFDoc.cxx @@ -172,6 +172,17 @@ Standard_GUID XCAFDoc::MaterialRefGUID () } +//======================================================================= +//function : NoteRefGUID +//purpose : +//======================================================================= + +Standard_GUID XCAFDoc::NoteRefGUID() +{ + static Standard_GUID ID ("F3599E50-F84A-493e-8D1B-1284E79322F1"); + return ID; +} + //======================================================================= //function : InvisibleGUID //purpose : @@ -239,6 +250,28 @@ Standard_GUID XCAFDoc::ViewRefPlaneGUID() return ID; } +//======================================================================= +//function : ViewRefPlaneGUID +//purpose : +//======================================================================= + +Standard_GUID XCAFDoc::ViewRefNoteGUID() +{ + static Standard_GUID ID("C814ACC6-43AC-4812-9B2A-4E9A2A549354"); + return ID; +} + +//======================================================================= +//function : ViewRefPlaneGUID +//purpose : +//======================================================================= + +Standard_GUID XCAFDoc::ViewRefAnnotationGUID() +{ + static Standard_GUID ID("A2B5BA42-DD00-43f5-8882-4B5F8E76B9D2"); + return ID; +} + //======================================================================= //function : LockGUID //purpose : diff --git a/src/XCAFDoc/XCAFDoc.hxx b/src/XCAFDoc/XCAFDoc.hxx index 1812029d9e..c59edef99d 100644 --- a/src/XCAFDoc/XCAFDoc.hxx +++ b/src/XCAFDoc/XCAFDoc.hxx @@ -97,7 +97,10 @@ public: Standard_EXPORT static Standard_GUID LayerRefGUID(); Standard_EXPORT static Standard_GUID MaterialRefGUID(); - + + //! Return GUIDs for representing notes + Standard_EXPORT static Standard_GUID NoteRefGUID(); + Standard_EXPORT static Standard_GUID InvisibleGUID(); //! Returns GUID for UAttribute identifying external reference on no-step file @@ -118,6 +121,10 @@ public: //! Return GUIDs for TreeNode representing specified types of View Standard_EXPORT static Standard_GUID ViewRefPlaneGUID(); + //! Return GUIDs for GraphNode representing specified types of View + Standard_EXPORT static Standard_GUID ViewRefNoteGUID(); + Standard_EXPORT static Standard_GUID ViewRefAnnotationGUID(); + //! Returns GUID for UAttribute identifying lock flag Standard_EXPORT static Standard_GUID LockGUID(); diff --git a/src/XCAFDoc/XCAFDoc_AssemblyItemId.cxx b/src/XCAFDoc/XCAFDoc_AssemblyItemId.cxx new file mode 100644 index 0000000000..29984514ca --- /dev/null +++ b/src/XCAFDoc/XCAFDoc_AssemblyItemId.cxx @@ -0,0 +1,123 @@ +// Created on: 2017-02-16 +// Created by: Sergey NIKONOV +// Copyright (c) 2000-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include + +XCAFDoc_AssemblyItemId::XCAFDoc_AssemblyItemId() +{ + +} + +XCAFDoc_AssemblyItemId::XCAFDoc_AssemblyItemId(const TColStd_ListOfAsciiString& thePath) +{ + Init(thePath); +} + +XCAFDoc_AssemblyItemId::XCAFDoc_AssemblyItemId(const TCollection_AsciiString& theString) +{ + Init(theString); +} + +void +XCAFDoc_AssemblyItemId::Init(const TColStd_ListOfAsciiString& thePath) +{ + myPath = thePath; +} + +void +XCAFDoc_AssemblyItemId::Init(const TCollection_AsciiString& theString) +{ + myPath.Clear(); + + for (Standard_Integer iEntry = 1; ; ++iEntry) + { + TCollection_AsciiString anEntry = theString.Token("/", iEntry); + if (anEntry.IsEmpty()) + break; + + myPath.Append(anEntry); + } +} + +Standard_Boolean +XCAFDoc_AssemblyItemId::IsNull() const +{ + return myPath.IsEmpty(); +} + +void +XCAFDoc_AssemblyItemId::Nullify() +{ + myPath.Clear(); +} + +Standard_Boolean +XCAFDoc_AssemblyItemId::IsChild(const XCAFDoc_AssemblyItemId& theOther) const +{ + if (myPath.Size() <= theOther.myPath.Size()) + return Standard_False; + + TColStd_ListOfAsciiString::Iterator anIt(myPath), anItOther(theOther.myPath); + for (; anItOther.More(); anIt.Next(), anItOther.Next()) + { + if (anIt.Value() != anItOther.Value()) + return Standard_False; + } + + return Standard_True; +} + +Standard_Boolean +XCAFDoc_AssemblyItemId::IsDirectChild(const XCAFDoc_AssemblyItemId& theOther) const +{ + return ((myPath.Size() == theOther.myPath.Size() - 1) && IsChild(theOther)); +} + +Standard_Boolean +XCAFDoc_AssemblyItemId::IsEqual(const XCAFDoc_AssemblyItemId& theOther) const +{ + if (this == &theOther) + return Standard_True; + + if (myPath.Size() != theOther.myPath.Size()) + return Standard_False; + + TColStd_ListOfAsciiString::Iterator anIt(myPath), anItOther(theOther.myPath); + for (; anIt.More() && anItOther.More(); anIt.Next(), anItOther.Next()) + { + if (anIt.Value() != anItOther.Value()) + return Standard_False; + } + + return Standard_True; +} + +const +TColStd_ListOfAsciiString& XCAFDoc_AssemblyItemId::GetPath() const +{ + return myPath; +} + +TCollection_AsciiString +XCAFDoc_AssemblyItemId::ToString() const +{ + TCollection_AsciiString aStr; + for (TColStd_ListOfAsciiString::Iterator anIt(myPath); anIt.More(); anIt.Next()) + { + aStr += '/'; + aStr += anIt.Value(); + } + return aStr; +} diff --git a/src/XCAFDoc/XCAFDoc_AssemblyItemId.hxx b/src/XCAFDoc/XCAFDoc_AssemblyItemId.hxx new file mode 100644 index 0000000000..94b8c040f3 --- /dev/null +++ b/src/XCAFDoc/XCAFDoc_AssemblyItemId.hxx @@ -0,0 +1,101 @@ +// Created on: 2017-02-16 +// Created by: Sergey NIKONOV +// Copyright (c) 2000-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _XCAFDoc_AssemblyItemId_HeaderFile +#define _XCAFDoc_AssemblyItemId_HeaderFile + +#include +#include + +//! Unique item identifier in the hierarchical product structure. +//! A full path to an assembly component in the “part-of” graph starting from +//! the root node. +class XCAFDoc_AssemblyItemId +{ + +public: + + //! Constructs an empty item ID. + Standard_EXPORT XCAFDoc_AssemblyItemId(); + + //! Constructs an item ID from a list of strings, where every + //! string is a label entry. + //! \param [in] thePath - list of label entries. + Standard_EXPORT XCAFDoc_AssemblyItemId(const TColStd_ListOfAsciiString& thePath); + + //! Constructs an item ID from a formatted path, where label entries + //! are separated by '/' symbol. + //! \param [in] theString - formatted full path. + Standard_EXPORT XCAFDoc_AssemblyItemId(const TCollection_AsciiString& theString); + + //! Initializes the item ID from a list of strings, where every + //! string is a label entry. + //! \param [in] thePath - list of label entries. + Standard_EXPORT void Init(const TColStd_ListOfAsciiString& thePath); + + //! Initializes the item ID from a formatted path, where label entries + //! are separated by '/' symbol. + //! \param [in] theString - formatted full path. + Standard_EXPORT void Init(const TCollection_AsciiString& theString); + + //! Returns true if the full path is empty, otherwise - false. + Standard_EXPORT Standard_Boolean IsNull() const; + + //! Clears the full path. + Standard_EXPORT void Nullify(); + + //! Checks if this item is a child of the given item. + //! \param [in] theOther - potentially ancestor item. + //! \return true if the item is a child of theOther item, otherwise - false. + Standard_EXPORT Standard_Boolean IsChild(const XCAFDoc_AssemblyItemId& theOther) const; + + //! Checks if this item is a direct child of the given item. + //! \param [in] theOther - potentially parent item. + //! \return true if the item is a direct child of theOther item, otherwise - false. + Standard_EXPORT Standard_Boolean IsDirectChild(const XCAFDoc_AssemblyItemId& theOther) const; + + //! Checks for item IDs equality. + //! \param [in] theOther - the item ID to check equality with. + //! \return true if this ID is equal to theOther, otherwise - false. + Standard_EXPORT Standard_Boolean IsEqual(const XCAFDoc_AssemblyItemId& theOther) const; + + //! Returns the full path as a list of label entries. + Standard_EXPORT const TColStd_ListOfAsciiString& GetPath() const; + + //! Returns the full pass as a formatted string. + Standard_EXPORT TCollection_AsciiString ToString() const; + + struct Hasher + { + static int HashCode(const XCAFDoc_AssemblyItemId& theItem, + const int upper) + { + return ::HashCode(theItem.ToString(), upper); + } + + static int IsEqual(const XCAFDoc_AssemblyItemId& theItem1, + const XCAFDoc_AssemblyItemId& theItem2) + { + return theItem1.IsEqual(theItem2); + } + }; + +private: + + TColStd_ListOfAsciiString myPath; ///< List of label entries + +}; + +#endif // _XCAFDoc_AssemblyItemId_HeaderFile diff --git a/src/XCAFDoc/XCAFDoc_AssemblyItemRef.cxx b/src/XCAFDoc/XCAFDoc_AssemblyItemRef.cxx new file mode 100644 index 0000000000..9059998e78 --- /dev/null +++ b/src/XCAFDoc/XCAFDoc_AssemblyItemRef.cxx @@ -0,0 +1,295 @@ +// Created on: 2017-02-16 +// Created by: Sergey NIKONOV +// Copyright (c) 2000-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +IMPLEMENT_STANDARD_RTTIEXT(XCAFDoc_AssemblyItemRef, TDF_Attribute) + +enum { + ExtraRef_None, + ExtraRef_AttrGUID, + ExtraRef_SubshapeIndex +}; + +const Standard_GUID& +XCAFDoc_AssemblyItemRef::GetID() +{ + static Standard_GUID s_ID("3F2E4CD6-169B-4747-A321-5670E4291F5D"); + return s_ID; +} + +Handle(XCAFDoc_AssemblyItemRef) +XCAFDoc_AssemblyItemRef::Get(const TDF_Label& theLabel) +{ + Handle(XCAFDoc_AssemblyItemRef) aThis; + theLabel.FindAttribute(XCAFDoc_AssemblyItemRef::GetID(), aThis); + return aThis; +} + +Handle(XCAFDoc_AssemblyItemRef) +XCAFDoc_AssemblyItemRef::Set(const TDF_Label& theLabel, + const XCAFDoc_AssemblyItemId& theItemId) +{ + Handle(XCAFDoc_AssemblyItemRef) aThis; + if (!theLabel.IsNull() && !theLabel.FindAttribute(XCAFDoc_AssemblyItemRef::GetID(), aThis)) + { + aThis = new XCAFDoc_AssemblyItemRef(); + aThis->SetItem(theItemId); + theLabel.AddAttribute(aThis); + } + return aThis; +} + +Handle(XCAFDoc_AssemblyItemRef) +XCAFDoc_AssemblyItemRef::Set(const TDF_Label& theLabel, + const XCAFDoc_AssemblyItemId& theItemId, + const Standard_GUID& theAttrGUID) +{ + Handle(XCAFDoc_AssemblyItemRef) aThis; + if (!theLabel.IsNull() && !theLabel.FindAttribute(XCAFDoc_AssemblyItemRef::GetID(), aThis)) + { + aThis = new XCAFDoc_AssemblyItemRef(); + aThis->SetItem(theItemId); + aThis->SetGUID(theAttrGUID); + theLabel.AddAttribute(aThis); + } + return aThis; +} + +Handle(XCAFDoc_AssemblyItemRef) +XCAFDoc_AssemblyItemRef::Set(const TDF_Label& theLabel, + const XCAFDoc_AssemblyItemId& theItemId, + const Standard_Integer theShapeIndex) +{ + Handle(XCAFDoc_AssemblyItemRef) aThis; + if (!theLabel.IsNull() && !theLabel.FindAttribute(XCAFDoc_AssemblyItemRef::GetID(), aThis)) + { + aThis = new XCAFDoc_AssemblyItemRef(); + aThis->SetItem(theItemId); + aThis->SetSubshapeIndex(theShapeIndex); + theLabel.AddAttribute(aThis); + } + return aThis; +} + +XCAFDoc_AssemblyItemRef::XCAFDoc_AssemblyItemRef() + : myExtraRef(ExtraRef_None) +{ + +} + +Standard_Boolean +XCAFDoc_AssemblyItemRef::IsOrphan() const +{ + if (myItemId.IsNull()) + return Standard_True; + + TDF_Label aRoot = Label().Root(); + + Handle(TDocStd_Owner) anOwner; + if (!aRoot.FindAttribute(TDocStd_Owner::GetID(), anOwner)) + return Standard_True; + + Handle(TDocStd_Document) aDoc = anOwner->GetDocument(); + if (aDoc.IsNull()) + return Standard_True; + + Handle(TDF_Data) aData = aDoc->GetData(); + if (aData.IsNull()) + return Standard_True; + + TDF_Label aLabel; + TDF_Tool::Label(aData, myItemId.GetPath().Last(), aLabel); + if (aLabel.IsNull()) + return Standard_True; + + if (HasExtraRef()) + { + if (IsGUID()) + { + Handle(TDF_Attribute) anAttr; + if (!aLabel.FindAttribute(GetGUID(), anAttr)) + return Standard_True; + } + else if (IsSubshapeIndex()) + { + Handle(TNaming_NamedShape) aNamedShape; + if (!aLabel.FindAttribute(TNaming_NamedShape::GetID(), aNamedShape)) + return Standard_True; + + TopoDS_Shape aShape = aNamedShape->Get(); + TopTools_IndexedMapOfShape aMap; + TopExp::MapShapes(aShape, aMap); + Standard_Integer aSubshapeIndex = GetSubshapeIndex(); + if (aSubshapeIndex < 1 || aMap.Size() < aSubshapeIndex) + return Standard_True; + } + } + + return Standard_False; +} + +Standard_Boolean +XCAFDoc_AssemblyItemRef::HasExtraRef() const +{ + return (myExtraRef != ExtraRef_None); +} + +Standard_Boolean +XCAFDoc_AssemblyItemRef::IsGUID() const +{ + return (myExtraRef == ExtraRef_AttrGUID && Standard_GUID::CheckGUIDFormat(myExtraId.ToCString())); +} + +Standard_Boolean +XCAFDoc_AssemblyItemRef::IsSubshapeIndex() const +{ + return (myExtraRef == ExtraRef_SubshapeIndex && myExtraId.IsIntegerValue()); +} + +const XCAFDoc_AssemblyItemId& +XCAFDoc_AssemblyItemRef::GetItem() const +{ + return myItemId; +} + +Standard_GUID +XCAFDoc_AssemblyItemRef::GetGUID() const +{ + if (IsGUID()) + return Standard_GUID(myExtraId.ToCString()); + else + return Standard_GUID(); +} + +Standard_Integer +XCAFDoc_AssemblyItemRef::GetSubshapeIndex() const +{ + if (IsSubshapeIndex()) + return myExtraId.IntegerValue(); + else + return 0; +} + +void +XCAFDoc_AssemblyItemRef::SetItem(const XCAFDoc_AssemblyItemId& theItemId) +{ + Backup(); + myItemId = theItemId; + ClearExtraRef(); +} + +void +XCAFDoc_AssemblyItemRef::SetItem(const TColStd_ListOfAsciiString& thePath) +{ + Backup(); + myItemId.Init(thePath); + ClearExtraRef(); +} + +void +XCAFDoc_AssemblyItemRef::SetItem(const TCollection_AsciiString& theString) +{ + Backup(); + myItemId.Init(theString); + ClearExtraRef(); +} + +void XCAFDoc_AssemblyItemRef::SetGUID(const Standard_GUID& theAttrGUID) +{ + Backup(); + myExtraRef = ExtraRef_AttrGUID; + Standard_Character aGUIDStr[Standard_GUID_SIZE + 1]; + theAttrGUID.ToCString(aGUIDStr); + aGUIDStr[Standard_GUID_SIZE] = '\0'; + myExtraId.Clear(); + myExtraId.AssignCat(aGUIDStr); +} + +void +XCAFDoc_AssemblyItemRef::SetSubshapeIndex(Standard_Integer theSubshapeIndex) +{ + Backup(); + myExtraRef = ExtraRef_SubshapeIndex; + myExtraId.Clear(); + myExtraId.AssignCat(theSubshapeIndex); +} + +void +XCAFDoc_AssemblyItemRef::ClearExtraRef() +{ + Backup(); + myExtraRef = ExtraRef_None; + myExtraId.Clear(); +} + +const Standard_GUID& +XCAFDoc_AssemblyItemRef::ID() const +{ + return GetID(); +} + +Handle(TDF_Attribute) +XCAFDoc_AssemblyItemRef::NewEmpty() const +{ + return new XCAFDoc_AssemblyItemRef(); +} + +void +XCAFDoc_AssemblyItemRef::Restore(const Handle(TDF_Attribute)& theAttrFrom) +{ + Handle(XCAFDoc_AssemblyItemRef) anOther = Handle(XCAFDoc_AssemblyItemRef)::DownCast(theAttrFrom); + if (!anOther.IsNull()) + { + myItemId = anOther->myItemId; + myExtraRef = anOther->myExtraRef; + myExtraId = anOther->myExtraId; + } +} + +void +XCAFDoc_AssemblyItemRef::Paste(const Handle(TDF_Attribute)& theAttrInto, + const Handle(TDF_RelocationTable)& /*theRT*/) const +{ + Handle(XCAFDoc_AssemblyItemRef) anOther = Handle(XCAFDoc_AssemblyItemRef)::DownCast(theAttrInto); + if (!anOther.IsNull()) + { + anOther->myItemId = myItemId; + anOther->myExtraRef = myExtraRef; + anOther->myExtraId = myExtraId; + } +} + +Standard_OStream& +XCAFDoc_AssemblyItemRef::Dump(Standard_OStream& theOS) const +{ + theOS << "Path: " << myItemId.ToString(); + if (IsGUID()) + theOS << "/GUID:" << myExtraId; + else if (IsSubshapeIndex()) + theOS << "/Subshape: " << myExtraId; + return theOS; +} diff --git a/src/XCAFDoc/XCAFDoc_AssemblyItemRef.hxx b/src/XCAFDoc/XCAFDoc_AssemblyItemRef.hxx new file mode 100644 index 0000000000..9f4eb60540 --- /dev/null +++ b/src/XCAFDoc/XCAFDoc_AssemblyItemRef.hxx @@ -0,0 +1,154 @@ +// Created on: 2017-02-16 +// Created by: Sergey NIKONOV +// Copyright (c) 2000-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _XCAFDoc_AssemblyItemRef_HeaderFile +#define _XCAFDoc_AssemblyItemRef_HeaderFile + +#include +#include +#include +#include +#include + +class TDF_Data; +class TDF_RelocationTable; + +class XCAFDoc_AssemblyItemRef; +DEFINE_STANDARD_HANDLE(XCAFDoc_AssemblyItemRef, TDF_Attribute) + +//! An attribute that describes a weak reference to an assembly item +//! or to a subshape or to an assembly label attribute. +class XCAFDoc_AssemblyItemRef : public TDF_Attribute +{ + +public: + + DEFINE_STANDARD_RTTIEXT(XCAFDoc_AssemblyItemRef, TDF_Attribute); + + Standard_EXPORT static const Standard_GUID& GetID(); + + //! Finds a reference attribute on the given label and returns it, if it is found + Standard_EXPORT static Handle(XCAFDoc_AssemblyItemRef) Get(const TDF_Label& theLabel); + + //! @name Set reference attribute functions. + //! @{ + + //! Create (if not exist) a reference to an assembly item. + //! \param [in] theLabel - label to add the attribute. + //! \param [in] theItemId - assembly item ID. + //! \return A handle to the attribute instance. + Standard_EXPORT static Handle(XCAFDoc_AssemblyItemRef) Set(const TDF_Label& theLabel, + const XCAFDoc_AssemblyItemId& theItemId); + + //! Create (if not exist) a reference to an assembly item's label attribute. + //! \param [in] theLabel - label to add the attribute. + //! \param [in] theItemId - assembly item ID. + //! \param [in] theGUID - assembly item's label attribute ID. + //! \return A handle to the attribute instance. + Standard_EXPORT static Handle(XCAFDoc_AssemblyItemRef) Set(const TDF_Label& theLabel, + const XCAFDoc_AssemblyItemId& theItemId, + const Standard_GUID& theGUID); + + //! Create (if not exist) a reference to an assembly item's subshape. + //! \param [in] theLabel - label to add the attribute. + //! \param [in] theItemId - assembly item ID. + //! \param [in] theShapeIndex - assembly item's subshape index. + //! \return A handle to the attribute instance. + Standard_EXPORT static Handle(XCAFDoc_AssemblyItemRef) Set(const TDF_Label& theLabel, + const XCAFDoc_AssemblyItemId& theItemId, + const Standard_Integer theShapeIndex); + + //! @} + + //! Creates an empty reference attribute. + Standard_EXPORT XCAFDoc_AssemblyItemRef(); + + //! Checks if the reference points to a really existing item in XDE document. + Standard_EXPORT Standard_Boolean IsOrphan() const; + + //! @name Extra reference functions. + //! @{ + + //! Checks if the reference points on an item's shapeindex or attribute. + Standard_EXPORT Standard_Boolean HasExtraRef() const; + + //! Checks is the reference points to an item's attribute. + Standard_EXPORT Standard_Boolean IsGUID() const; + + //! Checks is the reference points to an item's subshape. + Standard_EXPORT Standard_Boolean IsSubshapeIndex() const; + + //! Returns the assembly item's attribute that the reference points to. + //! If the reference doesn't point to an attribute, returns an empty GUID. + Standard_EXPORT Standard_GUID GetGUID() const; + + //! Returns the assembly item's subshape that the reference points to. + //! If the reference doesn't point to a subshape, returns 0. + Standard_EXPORT Standard_Integer GetSubshapeIndex() const; + + //! @} + + //! Returns the assembly item ID that the reference points to. + Standard_EXPORT const XCAFDoc_AssemblyItemId& GetItem() const; + + //! @name Set reference data functions. + //! @{ + + //! Sets the assembly item ID that the reference points to. + //! Extra reference data (if any) will be cleared. + Standard_EXPORT void SetItem(const XCAFDoc_AssemblyItemId& theItemId); + + //! Sets the assembly item ID from a list of label entries + //! that the reference points to. + //! Extra reference data (if any) will be cleared. + Standard_EXPORT void SetItem(const TColStd_ListOfAsciiString& thePath); + + //! Sets the assembly item ID from a formatted path + //! that the reference points to. + //! Extra reference data (if any) will be cleared. + Standard_EXPORT void SetItem(const TCollection_AsciiString& theString); + + //! Sets the assembly item's label attribute that the reference points to. + //! The base assembly item will not change. + Standard_EXPORT void SetGUID(const Standard_GUID& theAttrGUID); + + //! Sets the assembly item's subshape that the reference points to. + //! The base assembly item will not change. + Standard_EXPORT void SetSubshapeIndex(Standard_Integer theShapeIndex); + + //! @} + + //! Reverts the reference to empty state. + Standard_EXPORT void ClearExtraRef(); + +public: + + // Overrides TDF_Attribute pure virtuals + Standard_EXPORT const Standard_GUID& ID() const Standard_OVERRIDE; + Standard_EXPORT Handle(TDF_Attribute) NewEmpty() const Standard_OVERRIDE; + Standard_EXPORT void Restore(const Handle(TDF_Attribute)& theAttrFrom) Standard_OVERRIDE; + Standard_EXPORT void Paste(const Handle(TDF_Attribute)& theAttrInto, + const Handle(TDF_RelocationTable)& theRT) const Standard_OVERRIDE; + Standard_EXPORT Standard_OStream& Dump(Standard_OStream& theOS) const Standard_OVERRIDE; + +private: + + XCAFDoc_AssemblyItemId myItemId; ///< Assembly item ID + Standard_Integer myExtraRef; ///< Type of extra reference: subshape or attribute + TCollection_AsciiString myExtraId; ///< Extra reference data + +}; + +#endif // _XCAFDoc_AssemblyItemRef_HeaderFile diff --git a/src/XCAFDoc/XCAFDoc_DocumentTool.cxx b/src/XCAFDoc/XCAFDoc_DocumentTool.cxx index fdaa9c5448..5614c690f0 100644 --- a/src/XCAFDoc/XCAFDoc_DocumentTool.cxx +++ b/src/XCAFDoc/XCAFDoc_DocumentTool.cxx @@ -30,6 +30,7 @@ #include #include #include +#include #include #include @@ -83,6 +84,7 @@ Handle(XCAFDoc_DocumentTool) XCAFDoc_DocumentTool::Set(const TDF_Label& L, XCAFDoc_LayerTool::Set(LayersLabel(L)); XCAFDoc_DimTolTool::Set(DGTsLabel(L)); XCAFDoc_MaterialTool::Set(MaterialsLabel(L)); + XCAFDoc_NotesTool::Set(NotesLabel(L)); XCAFDoc_ViewTool::Set(ViewsLabel(L)); XCAFDoc_ClippingPlaneTool::Set(ClippingPlanesLabel(L)); } @@ -211,6 +213,18 @@ TDF_Label XCAFDoc_DocumentTool::ClippingPlanesLabel(const TDF_Label& acces) return L; } +//======================================================================= +//function : NotesLabel +//purpose : +//======================================================================= + +TDF_Label XCAFDoc_DocumentTool::NotesLabel(const TDF_Label& acces) +{ + TDF_Label L = DocLabel(acces).FindChild(9, Standard_True); + TDataStd_Name::Set(L, "Notes"); + return L; +} + //======================================================================= //function : ShapeTool //purpose : @@ -285,6 +299,16 @@ Handle(XCAFDoc_ClippingPlaneTool) XCAFDoc_DocumentTool::ClippingPlaneTool(const return XCAFDoc_ClippingPlaneTool::Set(ClippingPlanesLabel(acces)); } +//======================================================================= +//function : ClippingPlaneTool +//purpose : +//======================================================================= + +Handle(XCAFDoc_NotesTool) XCAFDoc_DocumentTool::NotesTool(const TDF_Label& acces) +{ + return XCAFDoc_NotesTool::Set(NotesLabel(acces)); +} + //======================================================================= //function : ID //purpose : diff --git a/src/XCAFDoc/XCAFDoc_DocumentTool.hxx b/src/XCAFDoc/XCAFDoc_DocumentTool.hxx index 5f467a4f4f..51a793fcca 100644 --- a/src/XCAFDoc/XCAFDoc_DocumentTool.hxx +++ b/src/XCAFDoc/XCAFDoc_DocumentTool.hxx @@ -30,6 +30,7 @@ class XCAFDoc_ClippingPlaneTool; class XCAFDoc_LayerTool; class XCAFDoc_DimTolTool; class XCAFDoc_MaterialTool; +class XCAFDoc_NotesTool; class XCAFDoc_ViewTool; class TDF_Attribute; class TDF_RelocationTable; @@ -84,7 +85,10 @@ public: //! Returns sub-label of DocLabel() with tag 8. Standard_EXPORT static TDF_Label ClippingPlanesLabel(const TDF_Label& acces); - + + //! Returns sub-label of DocLabel() with tag 9. + Standard_EXPORT static TDF_Label NotesLabel(const TDF_Label& acces); + //! Creates (if it does not exist) ShapeTool attribute on ShapesLabel(). Standard_EXPORT static Handle(XCAFDoc_ShapeTool) ShapeTool (const TDF_Label& acces); @@ -105,7 +109,10 @@ public: //! Creates (if it does not exist) ClippingPlaneTool attribute on ClippingPlanesLabel(). Standard_EXPORT static Handle(XCAFDoc_ClippingPlaneTool) ClippingPlaneTool(const TDF_Label& acces); - + + //! Creates (if it does not exist) NotesTool attribute on NotesLabel(). + Standard_EXPORT static Handle(XCAFDoc_NotesTool) NotesTool(const TDF_Label& acces); + Standard_EXPORT XCAFDoc_DocumentTool(); //! to be called when reading this attribute from file diff --git a/src/XCAFDoc/XCAFDoc_Note.cxx b/src/XCAFDoc/XCAFDoc_Note.cxx new file mode 100644 index 0000000000..f5a7d9e7d7 --- /dev/null +++ b/src/XCAFDoc/XCAFDoc_Note.cxx @@ -0,0 +1,103 @@ +// Created on: 2017-02-10 +// Created by: Sergey NIKONOV +// Copyright (c) 2000-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include +#include +#include +#include + +IMPLEMENT_STANDARD_RTTIEXT(XCAFDoc_Note, TDF_Attribute) + +Standard_Boolean +XCAFDoc_Note::IsMine(const TDF_Label& theLabel) +{ + return !Get(theLabel).IsNull(); +} + +XCAFDoc_Note::XCAFDoc_Note() +{ +} + +Handle(XCAFDoc_Note) +XCAFDoc_Note::Get(const TDF_Label& theLabel) +{ + Handle(XCAFDoc_Note) aNote; + for (TDF_AttributeIterator anIt(theLabel); anIt.More(); anIt.Next()) + { + aNote = Handle(XCAFDoc_Note)::DownCast(anIt.Value()); + if (!aNote.IsNull()) + break; + } + return aNote; +} + +void +XCAFDoc_Note::Set(const TCollection_ExtendedString& theUserName, + const TCollection_ExtendedString& theTimeStamp) +{ + Backup(); + + myUserName = theUserName; + myTimeStamp = theTimeStamp; +} + +const TCollection_ExtendedString& +XCAFDoc_Note::UserName() const +{ + return myUserName; +} + +const TCollection_ExtendedString& +XCAFDoc_Note::TimeStamp() const +{ + return myTimeStamp; +} + +Standard_Boolean +XCAFDoc_Note::IsOrphan() const +{ + Handle(XCAFDoc_GraphNode) aFather; + return !Label().FindAttribute(XCAFDoc::NoteRefGUID(), aFather) || + (aFather->NbChildren() == 0); +} + +void +XCAFDoc_Note::Restore(const Handle(TDF_Attribute)& theAttr) +{ + myUserName = Handle(XCAFDoc_Note)::DownCast(theAttr)->myUserName; + myTimeStamp = Handle(XCAFDoc_Note)::DownCast(theAttr)->myTimeStamp; +} + +void +XCAFDoc_Note::Paste(const Handle(TDF_Attribute)& theAttrInto, + const Handle(TDF_RelocationTable)& /*theRT*/) const +{ + Handle(XCAFDoc_Note)::DownCast(theAttrInto)->Set(myUserName, myTimeStamp); +} + +Standard_OStream& +XCAFDoc_Note::Dump(Standard_OStream& theOS) const +{ + TDF_Attribute::Dump(theOS); + theOS + << "Note : " + << (myUserName.IsEmpty() ? myUserName : "") + << " on " + << (myTimeStamp.IsEmpty() ? myTimeStamp : "") + ; + return theOS; +} diff --git a/src/XCAFDoc/XCAFDoc_Note.hxx b/src/XCAFDoc/XCAFDoc_Note.hxx new file mode 100644 index 0000000000..736e6f1802 --- /dev/null +++ b/src/XCAFDoc/XCAFDoc_Note.hxx @@ -0,0 +1,82 @@ +// Created on: 2017-02-10 +// Created by: Sergey NIKONOV +// Copyright (c) 2000-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _XCAFDoc_Note_HeaderFile +#define _XCAFDoc_Note_HeaderFile + +#include +#include +#include +#include +#include +#include + +class Standard_GUID; +class TDF_RelocationTable; + +class XCAFDoc_Note; +DEFINE_STANDARD_HANDLE(XCAFDoc_Note, TDF_Attribute) + +//! A base note attribute. +//! Any note contains the name of the user created the note +//! and the creation timestamp. +class XCAFDoc_Note : public TDF_Attribute +{ +public: + + DEFINE_STANDARD_RTTIEXT(XCAFDoc_Note, TDF_Attribute) + + //! Checks if the given label represents a note. + Standard_EXPORT static Standard_Boolean IsMine(const TDF_Label& theLabel); + + //! Finds a reference attribute on the given label and returns it, if it is found + Standard_EXPORT static Handle(XCAFDoc_Note) Get(const TDF_Label& theLabel); + + //! Sets the user name and the timestamp of the note. + //! \param [in] theUserName - the user associated with the note. + //! \param [in] theTimeStamp - timestamp of the note. + //! \return A handle to the attribute instance. + Standard_EXPORT void Set(const TCollection_ExtendedString& theUserName, + const TCollection_ExtendedString& theTimeStamp); + + //! Returns the user name, who created the note. + Standard_EXPORT const TCollection_ExtendedString& UserName() const; + + //! Returns the timestamp of the note. + Standard_EXPORT const TCollection_ExtendedString& TimeStamp() const; + + //! Checks if the note isn't linked to annotated items. + Standard_EXPORT Standard_Boolean IsOrphan() const; + +public: + + // Overrides TDF_Attribute virtuals + Standard_EXPORT void Restore(const Handle(TDF_Attribute)& theAttrFrom) Standard_OVERRIDE; + Standard_EXPORT void Paste(const Handle(TDF_Attribute)& theAttrInto, + const Handle(TDF_RelocationTable)& theRT) const Standard_OVERRIDE; + Standard_EXPORT Standard_OStream& Dump(Standard_OStream& theOS) const Standard_OVERRIDE; + +protected: + + //! Creates an empty note. + Standard_EXPORT XCAFDoc_Note(); + +private: + + TCollection_ExtendedString myUserName; ///< Name of the user, who created the note. + TCollection_ExtendedString myTimeStamp; ///< Timestamp, when the note was created. +}; + +#endif // _XCAFDoc_Note_HeaderFile diff --git a/src/XCAFDoc/XCAFDoc_NoteBalloon.cxx b/src/XCAFDoc/XCAFDoc_NoteBalloon.cxx new file mode 100644 index 0000000000..32288721b8 --- /dev/null +++ b/src/XCAFDoc/XCAFDoc_NoteBalloon.cxx @@ -0,0 +1,68 @@ +// Created on: 2017-08-10 +// Created by: Sergey NIKONOV +// Copyright (c) 2000-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include + +IMPLEMENT_STANDARD_RTTIEXT(XCAFDoc_NoteBalloon, XCAFDoc_NoteComment) + +const Standard_GUID& +XCAFDoc_NoteBalloon::GetID() +{ + static Standard_GUID s_ID("1127951D-87D5-4ecc-89D5-D1406576C43F"); + return s_ID; +} + +Handle(XCAFDoc_NoteBalloon) +XCAFDoc_NoteBalloon::Get(const TDF_Label& theLabel) +{ + Handle(XCAFDoc_NoteBalloon) aThis; + theLabel.FindAttribute(XCAFDoc_NoteBalloon::GetID(), aThis); + return aThis; +} + +Handle(XCAFDoc_NoteBalloon) +XCAFDoc_NoteBalloon::Set(const TDF_Label& theLabel, + const TCollection_ExtendedString& theUserName, + const TCollection_ExtendedString& theTimeStamp, + const TCollection_ExtendedString& theComment) +{ + Handle(XCAFDoc_NoteBalloon) aNoteBalloon; + if (!theLabel.IsNull() && !theLabel.FindAttribute(XCAFDoc_NoteBalloon::GetID(), aNoteBalloon)) + { + aNoteBalloon = new XCAFDoc_NoteBalloon(); + aNoteBalloon->XCAFDoc_Note::Set(theUserName, theTimeStamp); + aNoteBalloon->XCAFDoc_NoteComment::Set(theComment); + theLabel.AddAttribute(aNoteBalloon); + } + return aNoteBalloon; +} + +XCAFDoc_NoteBalloon::XCAFDoc_NoteBalloon() +{ +} + +const Standard_GUID& +XCAFDoc_NoteBalloon::ID() const +{ + return GetID(); +} + +Handle(TDF_Attribute) +XCAFDoc_NoteBalloon::NewEmpty() const +{ + return new XCAFDoc_NoteBalloon(); +} diff --git a/src/XCAFDoc/XCAFDoc_NoteBalloon.hxx b/src/XCAFDoc/XCAFDoc_NoteBalloon.hxx new file mode 100644 index 0000000000..10a77970a7 --- /dev/null +++ b/src/XCAFDoc/XCAFDoc_NoteBalloon.hxx @@ -0,0 +1,58 @@ +// Created on: 2017-08-10 +// Created by: Sergey NIKONOV +// Copyright (c) 2000-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _XCAFDoc_NoteBalloon_HeaderFile +#define _XCAFDoc_NoteBalloon_HeaderFile + +#include + +class XCAFDoc_NoteBalloon; +DEFINE_STANDARD_HANDLE(XCAFDoc_NoteBalloon, XCAFDoc_NoteComment) + +//! A comment note attribute. +//! Contains a textual comment. +class XCAFDoc_NoteBalloon : public XCAFDoc_NoteComment +{ +public: + + DEFINE_STANDARD_RTTIEXT(XCAFDoc_NoteBalloon, XCAFDoc_NoteComment) + + Standard_EXPORT static const Standard_GUID& GetID(); + + //! Finds a reference attribute on the given label and returns it, if it is found + Standard_EXPORT static Handle(XCAFDoc_NoteBalloon) Get(const TDF_Label& theLabel); + + //! Create (if not exist) a comment note on the given label. + //! \param [in] theLabel - note label. + //! \param [in] theUserName - the name of the user, who created the note. + //! \param [in] theTimeStamp - creation timestamp of the note. + //! \param [in] theComment - comment text. + Standard_EXPORT static Handle(XCAFDoc_NoteBalloon) Set(const TDF_Label& theLabel, + const TCollection_ExtendedString& theUserName, + const TCollection_ExtendedString& theTimeStamp, + const TCollection_ExtendedString& theComment); + + //! Creates an empty comment note. + Standard_EXPORT XCAFDoc_NoteBalloon(); + +public: + + // Overrides TDF_Attribute virtuals + Standard_EXPORT const Standard_GUID& ID() const Standard_OVERRIDE; + Standard_EXPORT Handle(TDF_Attribute) NewEmpty() const Standard_OVERRIDE; + +}; + +#endif // _XCAFDoc_NoteBalloon_HeaderFile diff --git a/src/XCAFDoc/XCAFDoc_NoteBinData.cxx b/src/XCAFDoc/XCAFDoc_NoteBinData.cxx new file mode 100644 index 0000000000..beb638f83d --- /dev/null +++ b/src/XCAFDoc/XCAFDoc_NoteBinData.cxx @@ -0,0 +1,195 @@ +// Created on: 2017-02-13 +// Created by: Sergey NIKONOV +// Copyright (c) 2000-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include +#include + +IMPLEMENT_STANDARD_RTTIEXT(XCAFDoc_NoteBinData, XCAFDoc_Note) + +const Standard_GUID& +XCAFDoc_NoteBinData::GetID() +{ + static Standard_GUID s_ID("E9055501-F0FC-4864-BE4B-284FDA7DDEAC"); + return s_ID; +} + +Handle(XCAFDoc_NoteBinData) +XCAFDoc_NoteBinData::Get(const TDF_Label& theLabel) +{ + Handle(XCAFDoc_NoteBinData) aThis; + theLabel.FindAttribute(XCAFDoc_NoteBinData::GetID(), aThis); + return aThis; +} + +Handle(XCAFDoc_NoteBinData) +XCAFDoc_NoteBinData::Set(const TDF_Label& theLabel, + const TCollection_ExtendedString& theUserName, + const TCollection_ExtendedString& theTimeStamp, + const TCollection_ExtendedString& theTitle, + const TCollection_AsciiString& theMIMEtype, + OSD_File& theFile) +{ + Handle(XCAFDoc_NoteBinData) aNoteBinData; + if (!theLabel.IsNull() && !theLabel.FindAttribute(XCAFDoc_NoteBinData::GetID(), aNoteBinData)) + { + aNoteBinData = new XCAFDoc_NoteBinData(); + aNoteBinData->XCAFDoc_Note::Set(theUserName, theTimeStamp); + if (aNoteBinData->Set(theTitle, theMIMEtype, theFile)) + theLabel.AddAttribute(aNoteBinData); + else + aNoteBinData.Nullify(); + } + return aNoteBinData; +} + +Handle(XCAFDoc_NoteBinData) +XCAFDoc_NoteBinData::Set(const TDF_Label& theLabel, + const TCollection_ExtendedString& theUserName, + const TCollection_ExtendedString& theTimeStamp, + const TCollection_ExtendedString& theTitle, + const TCollection_AsciiString& theMIMEtype, + const Handle(TColStd_HArray1OfByte)& theData) +{ + Handle(XCAFDoc_NoteBinData) aNoteBinData; + if (!theLabel.IsNull() && !theLabel.FindAttribute(XCAFDoc_NoteBinData::GetID(), aNoteBinData)) + { + aNoteBinData = new XCAFDoc_NoteBinData(); + aNoteBinData->XCAFDoc_Note::Set(theUserName, theTimeStamp); + aNoteBinData->Set(theTitle, theMIMEtype, theData); + theLabel.AddAttribute(aNoteBinData); + } + return aNoteBinData; +} + +XCAFDoc_NoteBinData::XCAFDoc_NoteBinData() +{ +} + +Standard_Boolean +XCAFDoc_NoteBinData::Set(const TCollection_ExtendedString& theTitle, + const TCollection_AsciiString& theMIMEtype, + OSD_File& theFile) +{ + if (!theFile.IsOpen() || !theFile.IsReadable()) + return Standard_False; + + Backup(); + + if (theFile.Size() > (Standard_Size)IntegerLast()) + return Standard_False; + + myData.reset(new TColStd_HArray1OfByte(1, (Standard_Integer)theFile.Size())); + Standard_Integer nbReadBytes = 0; + theFile.Read((Standard_Address)&myData->First(), myData->Length(), nbReadBytes); + if (nbReadBytes < myData->Length()) + return Standard_False; + + myTitle = theTitle; + myMIMEtype = theMIMEtype; + + return Standard_True; +} + +void +XCAFDoc_NoteBinData::Set(const TCollection_ExtendedString& theTitle, + const TCollection_AsciiString& theMIMEtype, + const Handle(TColStd_HArray1OfByte)& theData) +{ + Backup(); + + myData = theData; + myTitle = theTitle; + myMIMEtype = theMIMEtype; +} + +const TCollection_ExtendedString& +XCAFDoc_NoteBinData::Title() const +{ + return myTitle; +} + +const TCollection_AsciiString& +XCAFDoc_NoteBinData::MIMEtype() const +{ + return myMIMEtype; +} + +Standard_Integer +XCAFDoc_NoteBinData::Size() const +{ + return (!myData.IsNull() ? myData->Length() : 0); +} + +const Handle(TColStd_HArray1OfByte)& +XCAFDoc_NoteBinData::Data() const +{ + return myData; +} + +const +Standard_GUID& XCAFDoc_NoteBinData::ID() const +{ + return GetID(); +} + +Handle(TDF_Attribute) +XCAFDoc_NoteBinData::NewEmpty() const +{ + return new XCAFDoc_NoteBinData(); +} + +void +XCAFDoc_NoteBinData::Restore(const Handle(TDF_Attribute)& theAttr) +{ + XCAFDoc_Note::Restore(theAttr); + + Handle(XCAFDoc_NoteBinData) aMine = Handle(XCAFDoc_NoteBinData)::DownCast(theAttr); + if (!aMine.IsNull()) + { + myTitle = aMine->myTitle; + myMIMEtype = aMine->myMIMEtype; + myData = aMine->myData; + } +} + +void +XCAFDoc_NoteBinData::Paste(const Handle(TDF_Attribute)& theAttrInto, + const Handle(TDF_RelocationTable)& theRT) const +{ + XCAFDoc_Note::Paste(theAttrInto, theRT); + + Handle(XCAFDoc_NoteBinData) aMine = Handle(XCAFDoc_NoteBinData)::DownCast(theAttrInto); + if (!aMine.IsNull()) + aMine->Set(myTitle, myMIMEtype, myData); +} + +Standard_OStream& +XCAFDoc_NoteBinData::Dump(Standard_OStream& theOS) const +{ + XCAFDoc_Note::Dump(theOS); + theOS << "\n" + << "Title : " << (!myTitle.IsEmpty() ? myMIMEtype : "") << "\n" + << "MIME type : " << (!myMIMEtype.IsEmpty() ? myMIMEtype : "") << "\n" + << "Size : " << Size() << " bytes" << "\n" + ; + if (!myData.IsNull()) + { + for (Standard_Integer i = myData->Lower(); i <= myData->Upper(); ++i) + theOS << myData->Value(i); + } + return theOS; +} diff --git a/src/XCAFDoc/XCAFDoc_NoteBinData.hxx b/src/XCAFDoc/XCAFDoc_NoteBinData.hxx new file mode 100644 index 0000000000..dc902be3c8 --- /dev/null +++ b/src/XCAFDoc/XCAFDoc_NoteBinData.hxx @@ -0,0 +1,129 @@ +// Created on: 2017-02-13 +// Created by: Sergey NIKONOV +// Copyright (c) 2000-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _XCAFDoc_NoteBinData_HeaderFile +#define _XCAFDoc_NoteBinData_HeaderFile + +#include +#include +#include +#include + +class OSD_File; + +class XCAFDoc_NoteBinData; +DEFINE_STANDARD_HANDLE(XCAFDoc_NoteBinData, XCAFDoc_Note) + +class XCAFDoc_NoteBinData : public XCAFDoc_Note +{ +public: + + DEFINE_STANDARD_RTTIEXT(XCAFDoc_NoteBinData, XCAFDoc_Note) + + Standard_EXPORT static const Standard_GUID& GetID(); + + //! Finds a binary data attribute on the given label and returns it, if it is found + Standard_EXPORT static Handle(XCAFDoc_NoteBinData) Get(const TDF_Label& theLabel); + + //! @name Set attribute functions. + //! @{ + + //! Create (if not exist) a binary note with data loaded from a binary file. + //! \param [in] theLabel - label to add the attribute. + //! \param [in] theUserName - the name of the user, who created the note. + //! \param [in] theTimeStamp - creation timestamp of the note. + //! \param [in] theTitle - file title. + //! \param [in] theMIMEtype - MIME type of the file. + //! \param [in] theFile - input binary file. + //! \return A handle to the attribute instance. + Standard_EXPORT static Handle(XCAFDoc_NoteBinData) Set(const TDF_Label& theLabel, + const TCollection_ExtendedString& theUserName, + const TCollection_ExtendedString& theTimeStamp, + const TCollection_ExtendedString& theTitle, + const TCollection_AsciiString& theMIMEtype, + OSD_File& theFile); + + //! Create (if not exist) a binary note byte data array. + //! \param [in] theLabel - label to add the attribute. + //! \param [in] theUserName - the name of the user, who created the note. + //! \param [in] theTimeStamp - creation timestamp of the note. + //! \param [in] theTitle - data title. + //! \param [in] theMIMEtype - MIME type of data. + //! \param [in] theData - byte data array. + //! \return A handle to the attribute instance. + Standard_EXPORT static Handle(XCAFDoc_NoteBinData) Set(const TDF_Label& theLabel, + const TCollection_ExtendedString& theUserName, + const TCollection_ExtendedString& theTimeStamp, + const TCollection_ExtendedString& theTitle, + const TCollection_AsciiString& theMIMEtype, + const Handle(TColStd_HArray1OfByte)& theData); + + //! @} + + //! Creates an empty binary data note. + Standard_EXPORT XCAFDoc_NoteBinData(); + + //! @name Set attribute data functions. + //! @{ + + //! Sets title, MIME type and data from a binary file. + //! \param [in] theTitle - file title. + //! \param [in] theMIMEtype - MIME type of the file. + //! \param [in] theFile - input binary file. + Standard_EXPORT Standard_Boolean Set(const TCollection_ExtendedString& theTitle, + const TCollection_AsciiString& theMIMEtype, + OSD_File& theFile); + + //! Sets title, MIME type and data from a byte array. + //! \param [in] theTitle - data title. + //! \param [in] theMIMEtype - MIME type of data. + //! \param [in] theData - byte data array. + Standard_EXPORT void Set(const TCollection_ExtendedString& theTitle, + const TCollection_AsciiString& theMIMEtype, + const Handle(TColStd_HArray1OfByte)& theData); + + //! @} + + //! Returns the note title. + Standard_EXPORT const TCollection_ExtendedString& Title() const; + + //! Returns data MIME type. + Standard_EXPORT const TCollection_AsciiString& MIMEtype() const; + + //! Size of data in bytes. + Standard_EXPORT Standard_Integer Size() const; + + //! Returns byte data array. + Standard_EXPORT const Handle(TColStd_HArray1OfByte)& Data() const; + +public: + + // Overrides TDF_Attribute virtuals + Standard_EXPORT const Standard_GUID& ID() const Standard_OVERRIDE; + Standard_EXPORT Handle(TDF_Attribute) NewEmpty() const Standard_OVERRIDE; + Standard_EXPORT void Restore(const Handle(TDF_Attribute)& theAttrFrom) Standard_OVERRIDE; + Standard_EXPORT void Paste(const Handle(TDF_Attribute)& theAttrInto, + const Handle(TDF_RelocationTable)& theRT) const Standard_OVERRIDE; + Standard_EXPORT Standard_OStream& Dump(Standard_OStream& theOS) const Standard_OVERRIDE; + +protected: + + TCollection_ExtendedString myTitle; ///< Note title. + TCollection_AsciiString myMIMEtype; ///< MIME type of data. + Handle(TColStd_HArray1OfByte) myData; ///< Byte data array. + +}; + +#endif // _XCAFDoc_NoteBinData_HeaderFile diff --git a/src/XCAFDoc/XCAFDoc_NoteComment.cxx b/src/XCAFDoc/XCAFDoc_NoteComment.cxx new file mode 100644 index 0000000000..b88f721bb4 --- /dev/null +++ b/src/XCAFDoc/XCAFDoc_NoteComment.cxx @@ -0,0 +1,113 @@ +// Created on: 2017-02-13 +// Created by: Sergey NIKONOV +// Copyright (c) 2000-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include + +IMPLEMENT_STANDARD_RTTIEXT(XCAFDoc_NoteComment, XCAFDoc_Note) + +const Standard_GUID& +XCAFDoc_NoteComment::GetID() +{ + static Standard_GUID s_ID("FDEA4C52-0F54-484c-B590-579E18F7B5D4"); + return s_ID; +} + +Handle(XCAFDoc_NoteComment) +XCAFDoc_NoteComment::Get(const TDF_Label& theLabel) +{ + Handle(XCAFDoc_NoteComment) aThis; + theLabel.FindAttribute(XCAFDoc_NoteComment::GetID(), aThis); + return aThis; +} + +Handle(XCAFDoc_NoteComment) +XCAFDoc_NoteComment::Set(const TDF_Label& theLabel, + const TCollection_ExtendedString& theUserName, + const TCollection_ExtendedString& theTimeStamp, + const TCollection_ExtendedString& theComment) +{ + Handle(XCAFDoc_NoteComment) aNoteComment; + if (!theLabel.IsNull() && !theLabel.FindAttribute(XCAFDoc_NoteComment::GetID(), aNoteComment)) + { + aNoteComment = new XCAFDoc_NoteComment(); + aNoteComment->XCAFDoc_Note::Set(theUserName, theTimeStamp); + aNoteComment->Set(theComment); + theLabel.AddAttribute(aNoteComment); + } + return aNoteComment; +} + +XCAFDoc_NoteComment::XCAFDoc_NoteComment() +{ +} + +void +XCAFDoc_NoteComment::Set(const TCollection_ExtendedString& theComment) +{ + Backup(); + + myComment = theComment; +} + +const TCollection_ExtendedString& +XCAFDoc_NoteComment::Comment() const +{ + return myComment; +} + +const Standard_GUID& +XCAFDoc_NoteComment::ID() const +{ + return GetID(); +} + +Handle(TDF_Attribute) +XCAFDoc_NoteComment::NewEmpty() const +{ + return new XCAFDoc_NoteComment(); +} + +void +XCAFDoc_NoteComment::Restore(const Handle(TDF_Attribute)& theAttr) +{ + XCAFDoc_Note::Restore(theAttr); + + Handle(XCAFDoc_NoteComment) aMine = Handle(XCAFDoc_NoteComment)::DownCast(theAttr); + if (!aMine.IsNull()) + myComment = aMine->myComment; +} + +void +XCAFDoc_NoteComment::Paste(const Handle(TDF_Attribute)& theAttrInto, + const Handle(TDF_RelocationTable)& theRT) const +{ + XCAFDoc_Note::Paste(theAttrInto, theRT); + + Handle(XCAFDoc_NoteComment) aMine = Handle(XCAFDoc_NoteComment)::DownCast(theAttrInto); + if (!aMine.IsNull()) + aMine->Set(myComment); +} + +Standard_OStream& +XCAFDoc_NoteComment::Dump(Standard_OStream& theOS) const +{ + XCAFDoc_Note::Dump(theOS); + theOS << "\n" + << "Comment : " << (!myComment.IsEmpty() ? myComment : "") + ; + return theOS; +} diff --git a/src/XCAFDoc/XCAFDoc_NoteComment.hxx b/src/XCAFDoc/XCAFDoc_NoteComment.hxx new file mode 100644 index 0000000000..8232750698 --- /dev/null +++ b/src/XCAFDoc/XCAFDoc_NoteComment.hxx @@ -0,0 +1,72 @@ +// Created on: 2017-02-13 +// Created by: Sergey NIKONOV +// Copyright (c) 2000-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _XCAFDoc_NoteComment_HeaderFile +#define _XCAFDoc_NoteComment_HeaderFile + +#include + +class XCAFDoc_NoteComment; +DEFINE_STANDARD_HANDLE(XCAFDoc_NoteComment, XCAFDoc_Note) + +//! A comment note attribute. +//! Contains a textual comment. +class XCAFDoc_NoteComment : public XCAFDoc_Note +{ +public: + + DEFINE_STANDARD_RTTIEXT(XCAFDoc_NoteComment, XCAFDoc_Note) + + Standard_EXPORT static const Standard_GUID& GetID(); + + //! Finds a reference attribute on the given label and returns it, if it is found + Standard_EXPORT static Handle(XCAFDoc_NoteComment) Get(const TDF_Label& theLabel); + + //! Create (if not exist) a comment note on the given label. + //! \param [in] theLabel - note label. + //! \param [in] theUserName - the name of the user, who created the note. + //! \param [in] theTimeStamp - creation timestamp of the note. + //! \param [in] theComment - comment text. + Standard_EXPORT static Handle(XCAFDoc_NoteComment) Set(const TDF_Label& theLabel, + const TCollection_ExtendedString& theUserName, + const TCollection_ExtendedString& theTimeStamp, + const TCollection_ExtendedString& theComment); + + //! Creates an empty comment note. + Standard_EXPORT XCAFDoc_NoteComment(); + + //! Sets the comment text. + Standard_EXPORT void Set(const TCollection_ExtendedString& theComment); + + //! Returns the comment text. + Standard_EXPORT const TCollection_ExtendedString& Comment() const; + +public: + + // Overrides TDF_Attribute virtuals + Standard_EXPORT const Standard_GUID& ID() const Standard_OVERRIDE; + Standard_EXPORT Handle(TDF_Attribute) NewEmpty() const Standard_OVERRIDE; + Standard_EXPORT void Restore(const Handle(TDF_Attribute)& theAttrFrom) Standard_OVERRIDE; + Standard_EXPORT void Paste(const Handle(TDF_Attribute)& theAttrInto, + const Handle(TDF_RelocationTable)& theRT) const Standard_OVERRIDE; + Standard_EXPORT Standard_OStream& Dump(Standard_OStream& theOS) const Standard_OVERRIDE; + +protected: + + TCollection_ExtendedString myComment; ///< Comment text. + +}; + +#endif // _XCAFDoc_NoteComment_HeaderFile diff --git a/src/XCAFDoc/XCAFDoc_NotesTool.cxx b/src/XCAFDoc/XCAFDoc_NotesTool.cxx new file mode 100644 index 0000000000..96e8674643 --- /dev/null +++ b/src/XCAFDoc/XCAFDoc_NotesTool.cxx @@ -0,0 +1,848 @@ +// Created on: 2017-02-10 +// Created by: Sergey NIKONOV +// Copyright (c) 2000-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +namespace { + + XCAFDoc_AssemblyItemId labeledItem(const TDF_Label& theLabel) + { + TCollection_AsciiString anEntry; + TDF_Tool::Entry(theLabel, anEntry); + return XCAFDoc_AssemblyItemId(anEntry); + } + +} + +IMPLEMENT_STANDARD_RTTIEXT(XCAFDoc_NotesTool, TDF_Attribute) + +enum NotesTool_RootLabels +{ + NotesTool_NotesRoot = 1, + NotesTool_AnnotatedItemsRoot +}; + +const Standard_GUID& +XCAFDoc_NotesTool::GetID() +{ + static Standard_GUID s_ID("8F8174B1-6125-47a0-B357-61BD2D89380C"); + return s_ID; +} + +Handle(XCAFDoc_NotesTool) +XCAFDoc_NotesTool::Set(const TDF_Label& theLabel) +{ + Handle(XCAFDoc_NotesTool) aTool; + if (!theLabel.IsNull() && !theLabel.FindAttribute(XCAFDoc_NotesTool::GetID(), aTool)) + { + aTool = new XCAFDoc_NotesTool(); + theLabel.AddAttribute(aTool); + } + return aTool; +} + +XCAFDoc_NotesTool::XCAFDoc_NotesTool() +{ +} + +TDF_Label +XCAFDoc_NotesTool::GetNotesLabel() const +{ + return Label().FindChild(NotesTool_NotesRoot); +} + +TDF_Label XCAFDoc_NotesTool::GetAnnotatedItemsLabel() const +{ + return Label().FindChild(NotesTool_AnnotatedItemsRoot); +} + +Standard_Integer +XCAFDoc_NotesTool::NbNotes() const +{ + Standard_Integer nbNotes = 0; + for (TDF_ChildIterator anIter(GetNotesLabel()); anIter.More(); anIter.Next()) + { + const TDF_Label aLabel = anIter.Value(); + if (!XCAFDoc_Note::Get(aLabel).IsNull()) + ++nbNotes; + } + return nbNotes; +} + +Standard_Integer +XCAFDoc_NotesTool::NbAnnotatedItems() const +{ + Standard_Integer nbItems = 0; + for (TDF_ChildIDIterator anIter(GetAnnotatedItemsLabel(), XCAFDoc_AssemblyItemRef::GetID()); anIter.More(); anIter.Next()) + { + ++nbItems; + } + return nbItems; +} + +void +XCAFDoc_NotesTool::GetNotes(TDF_LabelSequence& theNoteLabels) const +{ + for (TDF_ChildIterator anIter(GetNotesLabel()); anIter.More(); anIter.Next()) + { + const TDF_Label aLabel = anIter.Value(); + if (!XCAFDoc_Note::Get(aLabel).IsNull()) + theNoteLabels.Append(aLabel); + } +} + +void +XCAFDoc_NotesTool::GetAnnotatedItems(TDF_LabelSequence& theItemLabels) const +{ + for (TDF_ChildIDIterator anIter(GetAnnotatedItemsLabel(), XCAFDoc_AssemblyItemRef::GetID()); anIter.More(); anIter.Next()) + { + theItemLabels.Append(anIter.Value()->Label()); + } +} + +Standard_Boolean +XCAFDoc_NotesTool::IsAnnotatedItem(const XCAFDoc_AssemblyItemId& theItemId) const +{ + return !FindAnnotatedItem(theItemId).IsNull(); +} + +Standard_Boolean +XCAFDoc_NotesTool::IsAnnotatedItem(const TDF_Label& theItemLabel) const +{ + return IsAnnotatedItem(labeledItem(theItemLabel)); +} + +TDF_Label +XCAFDoc_NotesTool::FindAnnotatedItem(const XCAFDoc_AssemblyItemId& theItemId) const +{ + for (TDF_ChildIDIterator anIter(GetAnnotatedItemsLabel(), XCAFDoc_AssemblyItemRef::GetID()); anIter.More(); anIter.Next()) + { + Handle(XCAFDoc_AssemblyItemRef) anItemRef = Handle(XCAFDoc_AssemblyItemRef)::DownCast(anIter.Value()); + if (!anItemRef.IsNull() && anItemRef->GetItem().IsEqual(theItemId) && !anItemRef->HasExtraRef()) + return anItemRef->Label(); + } + return TDF_Label(); +} + +TDF_Label +XCAFDoc_NotesTool::FindAnnotatedItem(const TDF_Label& theItemLabel) const +{ + return FindAnnotatedItem(labeledItem(theItemLabel)); +} + +TDF_Label +XCAFDoc_NotesTool::FindAnnotatedItemAttr(const XCAFDoc_AssemblyItemId& theItemId, + const Standard_GUID& theGUID) const +{ + for (TDF_ChildIDIterator anIter(GetAnnotatedItemsLabel(), XCAFDoc_AssemblyItemRef::GetID()); anIter.More(); anIter.Next()) + { + Handle(XCAFDoc_AssemblyItemRef) anItemRef = Handle(XCAFDoc_AssemblyItemRef)::DownCast(anIter.Value()); + if (!anItemRef.IsNull() && anItemRef->GetItem().IsEqual(theItemId) && + anItemRef->HasExtraRef() && anItemRef->GetGUID() == theGUID) + return anItemRef->Label(); + } + return TDF_Label(); +} + +TDF_Label +XCAFDoc_NotesTool::FindAnnotatedItemAttr(const TDF_Label& theItemLabel, + const Standard_GUID& theGUID) const +{ + return FindAnnotatedItemAttr(labeledItem(theItemLabel), theGUID); +} + +TDF_Label +XCAFDoc_NotesTool::FindAnnotatedItemSubshape(const XCAFDoc_AssemblyItemId& theItemId, + Standard_Integer theSubshapeIndex) const +{ + for (TDF_ChildIDIterator anIter(GetAnnotatedItemsLabel(), XCAFDoc_AssemblyItemRef::GetID()); anIter.More(); anIter.Next()) + { + Handle(XCAFDoc_AssemblyItemRef) anItemRef = Handle(XCAFDoc_AssemblyItemRef)::DownCast(anIter.Value()); + if (!anItemRef.IsNull() && anItemRef->GetItem().IsEqual(theItemId) && + anItemRef->HasExtraRef() && anItemRef->GetSubshapeIndex() == theSubshapeIndex) + return anItemRef->Label(); + } + return TDF_Label(); +} + +TDF_Label +XCAFDoc_NotesTool::FindAnnotatedItemSubshape(const TDF_Label& theItemLabel, + Standard_Integer theSubshapeIndex) const +{ + return FindAnnotatedItemSubshape(labeledItem(theItemLabel), theSubshapeIndex); +} + +Handle(XCAFDoc_Note) +XCAFDoc_NotesTool::CreateComment(const TCollection_ExtendedString& theUserName, + const TCollection_ExtendedString& theTimeStamp, + const TCollection_ExtendedString& theComment) +{ + TDF_Label aNoteLabel; + TDF_TagSource aTag; + aNoteLabel = aTag.NewChild(GetNotesLabel()); + return XCAFDoc_NoteComment::Set(aNoteLabel, theUserName, theTimeStamp, theComment); +} + +Handle(XCAFDoc_Note) +XCAFDoc_NotesTool::CreateBalloon(const TCollection_ExtendedString& theUserName, + const TCollection_ExtendedString& theTimeStamp, + const TCollection_ExtendedString& theComment) +{ + TDF_Label aNoteLabel; + TDF_TagSource aTag; + aNoteLabel = aTag.NewChild(GetNotesLabel()); + return XCAFDoc_NoteBalloon::Set(aNoteLabel, theUserName, theTimeStamp, theComment); +} + +Handle(XCAFDoc_Note) +XCAFDoc_NotesTool::CreateBinData(const TCollection_ExtendedString& theUserName, + const TCollection_ExtendedString& theTimeStamp, + const TCollection_ExtendedString& theTitle, + const TCollection_AsciiString& theMIMEtype, + OSD_File& theFile) +{ + TDF_Label aNoteLabel; + TDF_TagSource aTag; + aNoteLabel = aTag.NewChild(GetNotesLabel()); + return XCAFDoc_NoteBinData::Set(aNoteLabel, theUserName, theTimeStamp, theTitle, theMIMEtype, theFile); +} + +Handle(XCAFDoc_Note) +XCAFDoc_NotesTool::CreateBinData(const TCollection_ExtendedString& theUserName, + const TCollection_ExtendedString& theTimeStamp, + const TCollection_ExtendedString& theTitle, + const TCollection_AsciiString& theMIMEtype, + const Handle(TColStd_HArray1OfByte)& theData) +{ + TDF_Label aNoteLabel; + TDF_TagSource aTag; + aNoteLabel = aTag.NewChild(GetNotesLabel()); + return XCAFDoc_NoteBinData::Set(aNoteLabel, theUserName, theTimeStamp, theTitle, theMIMEtype, theData); +} + +Standard_Integer +XCAFDoc_NotesTool::GetNotes(const XCAFDoc_AssemblyItemId& theItemId, + TDF_LabelSequence& theNoteLabels) const +{ + TDF_Label anAnnotatedItem = FindAnnotatedItem(theItemId); + if (anAnnotatedItem.IsNull()) + return 0; + + Handle(XCAFDoc_GraphNode) aChild; + if (!anAnnotatedItem.FindAttribute(XCAFDoc::NoteRefGUID(), aChild)) + return 0; + + Standard_Integer nbFathers = aChild->NbFathers(); + for (Standard_Integer iFather = 1; iFather <= nbFathers; ++iFather) + { + Handle(XCAFDoc_GraphNode) aFather = aChild->GetFather(iFather); + theNoteLabels.Append(aFather->Label()); + } + + return theNoteLabels.Length(); +} + +Standard_Integer +XCAFDoc_NotesTool::GetNotes(const TDF_Label& theItemLabel, + TDF_LabelSequence& theNoteLabels) const +{ + return GetNotes(labeledItem(theItemLabel), theNoteLabels); +} + +Standard_Integer +XCAFDoc_NotesTool::GetAttrNotes(const XCAFDoc_AssemblyItemId& theItemId, + const Standard_GUID& theGUID, + TDF_LabelSequence& theNoteLabels) const +{ + TDF_Label anAnnotatedItem = FindAnnotatedItemAttr(theItemId, theGUID); + if (anAnnotatedItem.IsNull()) + return 0; + + Handle(XCAFDoc_GraphNode) aChild; + if (!anAnnotatedItem.FindAttribute(XCAFDoc::NoteRefGUID(), aChild)) + return 0; + + Standard_Integer nbFathers = aChild->NbFathers(); + for (Standard_Integer iFather = 1; iFather <= nbFathers; ++iFather) + { + Handle(XCAFDoc_GraphNode) aFather = aChild->GetFather(iFather); + theNoteLabels.Append(aFather->Label()); + } + + return theNoteLabels.Length(); +} + +Standard_Integer +XCAFDoc_NotesTool::GetAttrNotes(const TDF_Label& theItemLabel, + const Standard_GUID& theGUID, + TDF_LabelSequence& theNoteLabels) const +{ + return GetAttrNotes(labeledItem(theItemLabel), theGUID, theNoteLabels); +} + +Standard_Integer +XCAFDoc_NotesTool::GetSubshapeNotes(const XCAFDoc_AssemblyItemId& theItemId, + Standard_Integer theSubshapeIndex, + TDF_LabelSequence& theNoteLabels) const +{ + TDF_Label anAnnotatedItem = FindAnnotatedItemSubshape(theItemId, theSubshapeIndex); + if (anAnnotatedItem.IsNull()) + return 0; + + Handle(XCAFDoc_GraphNode) aChild; + if (!anAnnotatedItem.FindAttribute(XCAFDoc::NoteRefGUID(), aChild)) + return 0; + + Standard_Integer nbFathers = aChild->NbFathers(); + for (Standard_Integer iFather = 1; iFather <= nbFathers; ++iFather) + { + Handle(XCAFDoc_GraphNode) aFather = aChild->GetFather(iFather); + theNoteLabels.Append(aFather->Label()); + } + + return theNoteLabels.Length(); +} + +Handle(XCAFDoc_AssemblyItemRef) +XCAFDoc_NotesTool::AddNote(const TDF_Label& theNoteLabel, + const XCAFDoc_AssemblyItemId& theItemId) +{ + Handle(XCAFDoc_AssemblyItemRef) anItemRef; + + if (!XCAFDoc_Note::IsMine(theNoteLabel)) + return anItemRef; + + Handle(XCAFDoc_GraphNode) aChild; + TDF_Label anAnnotatedItem = FindAnnotatedItem(theItemId); + if (anAnnotatedItem.IsNull()) + { + TDF_TagSource aTag; + anAnnotatedItem = aTag.NewChild(GetAnnotatedItemsLabel()); + if (anAnnotatedItem.IsNull()) + return anItemRef; + } + + if (!anAnnotatedItem.FindAttribute(XCAFDoc::NoteRefGUID(), aChild)) + { + aChild = XCAFDoc_GraphNode::Set(anAnnotatedItem, XCAFDoc::NoteRefGUID()); + if (aChild.IsNull()) + return anItemRef; + } + + if (!anAnnotatedItem.FindAttribute(XCAFDoc_AssemblyItemRef::GetID(), anItemRef)) + { + anItemRef = XCAFDoc_AssemblyItemRef::Set(anAnnotatedItem, theItemId); + if (anItemRef.IsNull()) + return anItemRef; + } + + Handle(XCAFDoc_GraphNode) aFather; + if (!theNoteLabel.FindAttribute(XCAFDoc::NoteRefGUID(), aFather)) + { + aFather = XCAFDoc_GraphNode::Set(theNoteLabel, XCAFDoc::NoteRefGUID()); + if (aFather.IsNull()) + return anItemRef; + } + + aChild->SetFather(aFather); + aFather->SetChild(aChild); + + return anItemRef; +} + +Handle(XCAFDoc_AssemblyItemRef) +XCAFDoc_NotesTool::AddNote(const TDF_Label& theNoteLabel, + const TDF_Label& theItemLabel) +{ + return AddNote(theNoteLabel, labeledItem(theItemLabel)); +} + +Handle(XCAFDoc_AssemblyItemRef) +XCAFDoc_NotesTool::AddNoteToAttr(const TDF_Label& theNoteLabel, + const XCAFDoc_AssemblyItemId& theItemId, + const Standard_GUID& theGUID) +{ + Handle(XCAFDoc_AssemblyItemRef) anItemRef; + + if (!XCAFDoc_Note::IsMine(theNoteLabel)) + return anItemRef; + + Handle(XCAFDoc_GraphNode) aChild; + TDF_Label anAnnotatedItem = FindAnnotatedItemAttr(theItemId, theGUID); + if (anAnnotatedItem.IsNull()) + { + TDF_TagSource aTag; + anAnnotatedItem = aTag.NewChild(GetAnnotatedItemsLabel()); + if (anAnnotatedItem.IsNull()) + return anItemRef; + } + + if (!anAnnotatedItem.FindAttribute(XCAFDoc::NoteRefGUID(), aChild)) + { + aChild = XCAFDoc_GraphNode::Set(anAnnotatedItem, XCAFDoc::NoteRefGUID()); + if (aChild.IsNull()) + return anItemRef; + } + + if (!anAnnotatedItem.FindAttribute(XCAFDoc_AssemblyItemRef::GetID(), anItemRef)) + { + anItemRef = XCAFDoc_AssemblyItemRef::Set(anAnnotatedItem, theItemId); + if (anItemRef.IsNull()) + return anItemRef; + } + + Handle(XCAFDoc_GraphNode) aFather; + if (!theNoteLabel.FindAttribute(XCAFDoc::NoteRefGUID(), aFather)) + { + aFather = XCAFDoc_GraphNode::Set(theNoteLabel, XCAFDoc::NoteRefGUID()); + if (aFather.IsNull()) + return anItemRef; + } + + aChild->SetFather(aFather); + aFather->SetChild(aChild); + + anItemRef->SetGUID(theGUID); + + return anItemRef; +} + +Handle(XCAFDoc_AssemblyItemRef) +XCAFDoc_NotesTool::AddNoteToAttr(const TDF_Label& theNoteLabel, + const TDF_Label& theItemLabel, + const Standard_GUID& theGUID) +{ + return AddNoteToAttr(theNoteLabel, labeledItem(theItemLabel), theGUID); +} + +Handle(XCAFDoc_AssemblyItemRef) +XCAFDoc_NotesTool::AddNoteToSubshape(const TDF_Label& theNoteLabel, + const XCAFDoc_AssemblyItemId& theItemId, + Standard_Integer theSubshapeIndex) +{ + Handle(XCAFDoc_AssemblyItemRef) anItemRef; + + if (!XCAFDoc_Note::IsMine(theNoteLabel)) + return anItemRef; + + Handle(XCAFDoc_GraphNode) aChild; + TDF_Label anAnnotatedItem = FindAnnotatedItemSubshape(theItemId, theSubshapeIndex); + if (anAnnotatedItem.IsNull()) + { + TDF_TagSource aTag; + anAnnotatedItem = aTag.NewChild(GetAnnotatedItemsLabel()); + if (anAnnotatedItem.IsNull()) + return anItemRef; + } + + if (!anAnnotatedItem.FindAttribute(XCAFDoc::NoteRefGUID(), aChild)) + { + aChild = XCAFDoc_GraphNode::Set(anAnnotatedItem, XCAFDoc::NoteRefGUID()); + if (aChild.IsNull()) + return anItemRef; + } + + if (!anAnnotatedItem.FindAttribute(XCAFDoc_AssemblyItemRef::GetID(), anItemRef)) + { + anItemRef = XCAFDoc_AssemblyItemRef::Set(anAnnotatedItem, theItemId); + if (anItemRef.IsNull()) + return anItemRef; + } + + Handle(XCAFDoc_GraphNode) aFather; + if (!theNoteLabel.FindAttribute(XCAFDoc::NoteRefGUID(), aFather)) + { + aFather = XCAFDoc_GraphNode::Set(theNoteLabel, XCAFDoc::NoteRefGUID()); + if (aFather.IsNull()) + return anItemRef; + } + + aChild->SetFather(aFather); + aFather->SetChild(aChild); + + anItemRef->SetSubshapeIndex(theSubshapeIndex); + + return anItemRef; +} + +Handle(XCAFDoc_AssemblyItemRef) +XCAFDoc_NotesTool::AddNoteToSubshape(const TDF_Label& theNoteLabel, + const TDF_Label& theItemLabel, + Standard_Integer theSubshapeIndex) +{ + return AddNoteToSubshape(theNoteLabel, labeledItem(theItemLabel), theSubshapeIndex); +} + +Standard_Boolean +XCAFDoc_NotesTool::RemoveNote(const TDF_Label& theNoteLabel, + const XCAFDoc_AssemblyItemId& theItemId, + Standard_Boolean theDelIfOrphan) +{ + Handle(XCAFDoc_Note) aNote = XCAFDoc_Note::Get(theNoteLabel); + + if (aNote.IsNull()) + return Standard_False; + + Handle(XCAFDoc_GraphNode) aFather; + if (!theNoteLabel.FindAttribute(XCAFDoc::NoteRefGUID(), aFather)) + return Standard_False; + + TDF_Label anAnnotatedItem = FindAnnotatedItem(theItemId); + if (anAnnotatedItem.IsNull()) + return Standard_False; + + Handle(XCAFDoc_GraphNode) aChild; + if (!anAnnotatedItem.FindAttribute(XCAFDoc::NoteRefGUID(), aChild)) + return Standard_False; + + aChild->UnSetFather(aFather); + if (aChild->NbFathers() == 0) + anAnnotatedItem.ForgetAllAttributes(); + + if (theDelIfOrphan && aNote->IsOrphan()) + DeleteNote(theNoteLabel); + + return Standard_True; +} + +Standard_Boolean +XCAFDoc_NotesTool::RemoveNote(const TDF_Label& theNoteLabel, + const TDF_Label& theItemLabel, + Standard_Boolean theDelIfOrphan) +{ + return RemoveNote(theNoteLabel, labeledItem(theItemLabel), theDelIfOrphan); +} + +Standard_Boolean +XCAFDoc_NotesTool::RemoveSubshapeNote(const TDF_Label& theNoteLabel, + const XCAFDoc_AssemblyItemId& theItemId, + Standard_Integer theSubshapeIndex, + Standard_Boolean theDelIfOrphan) +{ + Handle(XCAFDoc_Note) aNote = XCAFDoc_Note::Get(theNoteLabel); + + if (aNote.IsNull()) + return Standard_False; + + Handle(XCAFDoc_GraphNode) aFather; + if (!theNoteLabel.FindAttribute(XCAFDoc::NoteRefGUID(), aFather)) + return Standard_False; + + TDF_Label anAnnotatedItem = FindAnnotatedItemSubshape(theItemId, theSubshapeIndex); + if (anAnnotatedItem.IsNull()) + return Standard_False; + + Handle(XCAFDoc_GraphNode) aChild; + if (!anAnnotatedItem.FindAttribute(XCAFDoc::NoteRefGUID(), aChild)) + return Standard_False; + + aChild->UnSetFather(aFather); + if (aChild->NbFathers() == 0) + anAnnotatedItem.ForgetAllAttributes(); + + if (theDelIfOrphan && aNote->IsOrphan()) + DeleteNote(theNoteLabel); + + return Standard_True; +} + +Standard_Boolean +XCAFDoc_NotesTool::RemoveSubshapeNote(const TDF_Label& theNoteLabel, + const TDF_Label& theItemLabel, + Standard_Integer theSubshapeIndex, + Standard_Boolean theDelIfOrphan) +{ + return RemoveSubshapeNote(theNoteLabel, labeledItem(theItemLabel), theSubshapeIndex, theDelIfOrphan); +} + +Standard_Boolean +XCAFDoc_NotesTool::RemoveAttrNote(const TDF_Label& theNoteLabel, + const XCAFDoc_AssemblyItemId& theItemId, + const Standard_GUID& theGUID, + Standard_Boolean theDelIfOrphan) +{ + Handle(XCAFDoc_Note) aNote = XCAFDoc_Note::Get(theNoteLabel); + + if (aNote.IsNull()) + return Standard_False; + + Handle(XCAFDoc_GraphNode) aFather; + if (!theNoteLabel.FindAttribute(XCAFDoc::NoteRefGUID(), aFather)) + return Standard_False; + + TDF_Label anAnnotatedItem = FindAnnotatedItemAttr(theItemId, theGUID); + if (anAnnotatedItem.IsNull()) + return Standard_False; + + Handle(XCAFDoc_GraphNode) aChild; + if (!anAnnotatedItem.FindAttribute(XCAFDoc::NoteRefGUID(), aChild)) + return Standard_False; + + aChild->UnSetFather(aFather); + if (aChild->NbFathers() == 0) + anAnnotatedItem.ForgetAllAttributes(); + + if (theDelIfOrphan && aNote->IsOrphan()) + DeleteNote(theNoteLabel); + + return Standard_True; +} + +Standard_Boolean +XCAFDoc_NotesTool::RemoveAttrNote(const TDF_Label& theNoteLabel, + const TDF_Label& theItemLabel, + const Standard_GUID& theGUID, + Standard_Boolean theDelIfOrphan) +{ + return RemoveAttrNote(theNoteLabel, labeledItem(theItemLabel), theGUID, theDelIfOrphan); +} + +Standard_Boolean +XCAFDoc_NotesTool::RemoveAllNotes(const XCAFDoc_AssemblyItemId& theItemId, + Standard_Boolean theDelIfOrphan) +{ + TDF_Label anAnnotatedItem = FindAnnotatedItem(theItemId); + if (anAnnotatedItem.IsNull()) + return Standard_False; + + Handle(XCAFDoc_GraphNode) aChild; + if (!anAnnotatedItem.FindAttribute(XCAFDoc::NoteRefGUID(), aChild)) + return Standard_False; + + while (aChild->NbFathers() > 0) + { + Handle(XCAFDoc_GraphNode) aFather = aChild->GetFather(1); + Handle(XCAFDoc_Note) aNote = XCAFDoc_Note::Get(aFather->Label()); + if (!aNote.IsNull()) + { + aFather->UnSetChild(aChild); + if (theDelIfOrphan && aNote->IsOrphan()) + DeleteNote(aFather->Label()); + } + } + + anAnnotatedItem.ForgetAllAttributes(); + + return Standard_True; +} + +Standard_Boolean +XCAFDoc_NotesTool::RemoveAllNotes(const TDF_Label& theItemLabel, + Standard_Boolean theDelIfOrphan) +{ + return RemoveAllNotes(labeledItem(theItemLabel), theDelIfOrphan); +} + +Standard_Boolean +XCAFDoc_NotesTool::RemoveAllSubshapeNotes(const XCAFDoc_AssemblyItemId& theItemId, + Standard_Integer theSubshapeIndex, + Standard_Boolean theDelIfOrphan) +{ + TDF_Label anAnnotatedItem = FindAnnotatedItemSubshape(theItemId, theSubshapeIndex); + if (anAnnotatedItem.IsNull()) + return Standard_False; + + Handle(XCAFDoc_GraphNode) aChild; + if (!anAnnotatedItem.FindAttribute(XCAFDoc::NoteRefGUID(), aChild)) + return Standard_False; + + while (aChild->NbFathers() > 0) + { + Handle(XCAFDoc_GraphNode) aFather = aChild->GetFather(1); + Handle(XCAFDoc_Note) aNote = XCAFDoc_Note::Get(aFather->Label()); + if (!aNote.IsNull()) + { + aFather->UnSetChild(aChild); + if (theDelIfOrphan && aNote->IsOrphan()) + DeleteNote(aFather->Label()); + } + } + + anAnnotatedItem.ForgetAllAttributes(); + + return Standard_True; +} + +Standard_Boolean +XCAFDoc_NotesTool::RemoveAllAttrNotes(const XCAFDoc_AssemblyItemId& theItemId, + const Standard_GUID& theGUID, + Standard_Boolean theDelIfOrphan) +{ + TDF_Label anAnnotatedItem = FindAnnotatedItemAttr(theItemId, theGUID); + if (anAnnotatedItem.IsNull()) + return Standard_False; + + Handle(XCAFDoc_GraphNode) aChild; + if (!anAnnotatedItem.FindAttribute(XCAFDoc::NoteRefGUID(), aChild)) + return Standard_False; + + while (aChild->NbFathers() > 0) + { + Handle(XCAFDoc_GraphNode) aFather = aChild->GetFather(1); + Handle(XCAFDoc_Note) aNote = XCAFDoc_Note::Get(aFather->Label()); + if (!aNote.IsNull()) + { + aFather->UnSetChild(aChild); + if (theDelIfOrphan && aNote->IsOrphan()) + DeleteNote(aFather->Label()); + } + } + + anAnnotatedItem.ForgetAllAttributes(); + + return Standard_True; +} + +Standard_Boolean +XCAFDoc_NotesTool::RemoveAllAttrNotes(const TDF_Label& theItemLabel, + const Standard_GUID& theGUID, + Standard_Boolean theDelIfOrphan) +{ + return RemoveAllAttrNotes(labeledItem(theItemLabel), theGUID, theDelIfOrphan); +} + +Standard_Boolean +XCAFDoc_NotesTool::DeleteNote(const TDF_Label& theNoteLabel) +{ + Handle(XCAFDoc_Note) aNote = XCAFDoc_Note::Get(theNoteLabel); + if (!aNote.IsNull()) + { + Handle(XCAFDoc_GraphNode) aFather; + if (theNoteLabel.FindAttribute(XCAFDoc::NoteRefGUID(), aFather) && !aFather.IsNull()) + { + while (aFather->NbChildren() > 0) + { + Handle(XCAFDoc_GraphNode) aChild = aFather->GetChild(1); + aFather->UnSetChild(aChild); + if (aChild->NbFathers() == 0) + aChild->Label().ForgetAllAttributes(Standard_True); + } + } + theNoteLabel.ForgetAllAttributes(Standard_True); + return Standard_True; + } + return Standard_False; +} + +Standard_Integer +XCAFDoc_NotesTool::DeleteNotes(TDF_LabelSequence& theNoteLabels) +{ + Standard_Integer nbNotes = 0; + for (TDF_LabelSequence::Iterator anIter(theNoteLabels); anIter.More(); anIter.Next()) + { + if (DeleteNote(anIter.Value())) + ++nbNotes; + } + return nbNotes; +} + +Standard_Integer +XCAFDoc_NotesTool::DeleteAllNotes() +{ + Standard_Integer nbNotes = 0; + for (TDF_ChildIterator anIter(GetNotesLabel()); anIter.More(); anIter.Next()) + { + if (DeleteNote(anIter.Value())) + ++nbNotes; + } + return nbNotes; +} + +Standard_Integer +XCAFDoc_NotesTool::NbOrphanNotes() const +{ + Standard_Integer nbNotes = 0; + for (TDF_ChildIterator anIter(GetNotesLabel()); anIter.More(); anIter.Next()) + { + const TDF_Label aLabel = anIter.Value(); + Handle(XCAFDoc_Note) aNote = XCAFDoc_Note::Get(aLabel); + if (!aNote.IsNull() && aNote->IsOrphan()) + ++nbNotes; + } + return nbNotes; +} + +void +XCAFDoc_NotesTool::GetOrphanNotes(TDF_LabelSequence& theNoteLabels) const +{ + for (TDF_ChildIterator anIter(GetNotesLabel()); anIter.More(); anIter.Next()) + { + const TDF_Label aLabel = anIter.Value(); + Handle(XCAFDoc_Note) aNote = XCAFDoc_Note::Get(aLabel); + if (!aNote.IsNull() && aNote->IsOrphan()) + theNoteLabels.Append(aLabel); + } +} + +Standard_Integer +XCAFDoc_NotesTool::DeleteOrphanNotes() +{ + Standard_Integer nbNotes = 0; + for (TDF_ChildIterator anIter(GetNotesLabel()); anIter.More(); anIter.Next()) + { + const TDF_Label aLabel = anIter.Value(); + Handle(XCAFDoc_Note) aNote = XCAFDoc_Note::Get(aLabel); + if (!aNote.IsNull() && aNote->IsOrphan() && DeleteNote(aLabel)) + ++nbNotes; + } + return nbNotes; +} + +const Standard_GUID& +XCAFDoc_NotesTool::ID() const +{ + return GetID(); +} + +Handle(TDF_Attribute) +XCAFDoc_NotesTool::NewEmpty() const +{ + return new XCAFDoc_NotesTool(); +} + +void +XCAFDoc_NotesTool::Restore(const Handle(TDF_Attribute)& /*theAttr*/) +{ +} + +void +XCAFDoc_NotesTool::Paste(const Handle(TDF_Attribute)& /*theAttrInto*/, + const Handle(TDF_RelocationTable)& /*theRT*/) const +{ +} + +Standard_OStream& +XCAFDoc_NotesTool::Dump(Standard_OStream& theOS) const +{ + theOS + << "Notes : " << NbNotes() << "\n" + << "Annotated items : " << NbAnnotatedItems() << "\n" + ; + return theOS; +} diff --git a/src/XCAFDoc/XCAFDoc_NotesTool.hxx b/src/XCAFDoc/XCAFDoc_NotesTool.hxx new file mode 100644 index 0000000000..b3cf26beeb --- /dev/null +++ b/src/XCAFDoc/XCAFDoc_NotesTool.hxx @@ -0,0 +1,519 @@ +// Created on: 2017-02-10 +// Created by: Sergey NIKONOV +// Copyright (c) 2000-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _XCAFDoc_NotesTool_HeaderFile +#define _XCAFDoc_NotesTool_HeaderFile + +#include +#include +#include +#include + +class OSD_File; +class Standard_GUID; +class TCollection_AsciiString; +class TCollection_ExtendedString; +class TColStd_HArray1OfByte; +class TDF_RelocationTable; +class XCAFDoc_Note; +class XCAFDoc_AssemblyItemId; +class XCAFDoc_AssemblyItemRef; + +class XCAFDoc_NotesTool; +DEFINE_STANDARD_HANDLE(XCAFDoc_NotesTool, TDF_Attribute) + +//! A tool to annotate items in the hierarchical product structure. +//! There are two basic entities, which operates the notes tool: notes +//! and annotated items. A note is a user defined data structure derived +//! from \ref XCAFDoc_Note attribute that is attached to a separate label under +//! the notes hive. An annotated item is represented by \ref XCAFDoc_AssemblyItemRef +//! attribute attached to a separate label under the annotated items +//! hive. Notes are linked with annotated items by means of \ref XCAFDoc_GraphNode +//! attribute. Notes play parent roles and annotated items - child roles. +//! +//! ------------------------ +//! | XCAFDoc_DocumentTool | +//! | 0:1 | +//! ------------------------ +//! |1 +//! ------------------------ +//! | XCAFDoc_NotesTool | +//! | 0:1:9 | +//! ------------------------ +//! |1 +//! | ------------------- --------------------------- +//! +___| Notes |-----| XCAFDoc_Note | +//! | 1| 0:1:9:1 |1 *| 0:1:9:1:* | +//! | ------------------- --------------------------- +//! | !* +//! | { XCAFDoc_GraphNode } +//! | *! +//! | ------------------- --------------------------- +//! +___| Annotated items |-----| XCAFDoc_AssemblyItemRef | +//! 1| 0:1:9:2 |1 *| 0:1:9:2:* | +//! ------------------- --------------------------- +//! +//! A typical annotation procedure is illustrated by the code example below: +//! \code{.c++} +//! // Get the notes tool from a XCAF document +//! Handle(XCAFDoc_NotesTool) aNotesTool = XCAFDoc_DocumentTool::NotesTool(aDoc->Main()); +//! // Create new comment note +//! Handle(XCAFDoc_Note) aNote = aNotesTool->CreateComment(aUserName, aTimestamp, aComment); +//! if (!aNote.IsNull()) { +//! Handle(XCAFDoc_AssemblyItemRef) aRef = aNotesTool->AddNote(aNote->Label(), anAssemblyItemId); +//! if (aRef.IsNull()) { +//! // Process error... +//! } +//! } +//! \endcode +class XCAFDoc_NotesTool : public TDF_Attribute +{ +public: + + DEFINE_STANDARD_RTTIEXT(XCAFDoc_NotesTool, TDF_Attribute) + + Standard_EXPORT static const Standard_GUID& GetID(); + + //! Create (if not exist) a notes tool from XCAFDoc on theLabel. + Standard_EXPORT static Handle(XCAFDoc_NotesTool) Set(const TDF_Label& theLabel); + + //! Creates an empty notes tool. + Standard_EXPORT XCAFDoc_NotesTool(); + + //! Returns the label of the notes hive. + Standard_EXPORT TDF_Label GetNotesLabel() const; + + //! Returns the label of the annotated items hive. + Standard_EXPORT TDF_Label GetAnnotatedItemsLabel() const; + + //! Returns the number of labels in the notes hive. + Standard_EXPORT Standard_Integer NbNotes() const; + + //! Returns the number of labels in the annotated items hive. + Standard_EXPORT Standard_Integer NbAnnotatedItems() const; + + //! Returns all labels from the notes hive. + //! The label sequence isn't cleared beforehand. + //! \param [out] theNoteLabels - sequence of labels. + Standard_EXPORT void GetNotes(TDF_LabelSequence& theNoteLabels) const; + + //! Returns all labels from the annotated items hive. + //! The label sequence isn't cleared beforehand. + //! \param [out] theNoteLabels - sequence of labels. + Standard_EXPORT void GetAnnotatedItems(TDF_LabelSequence& theLabels) const; + + //! Checks if the given assembly item is annotated. + //! \param [in] theItemId - assembly item ID. + //! \return true if the item is annotated, otherwise - false. + Standard_EXPORT Standard_Boolean IsAnnotatedItem(const XCAFDoc_AssemblyItemId& theItemId) const; + + //! Checks if the given labeled item is annotated. + //! \param [in] theItemLabel - item label. + //! \return true if the item is annotated, otherwise - false. + Standard_EXPORT Standard_Boolean IsAnnotatedItem(const TDF_Label& theItemLabel) const; + + //! @name Find annotated item functions + //! @{ + + //! Finds a label of the given assembly item ID in the annotated items hive. + //! \param [in] theItemId - assembly item ID. + //! \return annotated item label if it is found, otherwise - null label. + Standard_EXPORT TDF_Label FindAnnotatedItem(const XCAFDoc_AssemblyItemId& theItemId) const; + + //! Finds a label of the given labeled item in the annotated items hive. + //! \param [in] theItemLabel - item label. + //! \return annotated item label if it is found, otherwise - null label. + Standard_EXPORT TDF_Label FindAnnotatedItem(const TDF_Label& theItemLabel) const; + + //! Finds a label of the given assembly item's attribute in the annotated items hive. + //! \param [in] theItemId - assembly item ID. + //! \param [in] theGUID - assembly item's attribute GUID. + //! \return annotated item label if it is found, otherwise - null label. + Standard_EXPORT TDF_Label FindAnnotatedItemAttr(const XCAFDoc_AssemblyItemId& theItemId, + const Standard_GUID& theGUID) const; + + //! Finds a label of the given labeled item's attribute in the annotated items hive. + //! \param [in] theItemLabel - item label. + //! \param [in] theGUID - item's attribute GUID. + //! \return annotated item label if it is found, otherwise - null label. + Standard_EXPORT TDF_Label FindAnnotatedItemAttr(const TDF_Label& theItemLabel, + const Standard_GUID& theGUID) const; + + //! Finds a label of the given assembly item's subshape in the annotated items hive. + //! \param [in] theItemId - assembly item ID. + //! \param [in] theSubshapeIndex - assembly item's subshape index. + //! \return annotated item label if it is found, otherwise - null label. + Standard_EXPORT TDF_Label FindAnnotatedItemSubshape(const XCAFDoc_AssemblyItemId& theItemId, + Standard_Integer theSubshapeIndex) const; + + //! Finds a label of the given labeled item's subshape in the annotated items hive. + //! \param [in] theItemLabel - item label. + //! \param [in] theSubshapeIndex - labeled item's subshape index. + //! \return annotated item label if it is found, otherwise - null label. + Standard_EXPORT TDF_Label FindAnnotatedItemSubshape(const TDF_Label& theItemLabel, + Standard_Integer theSubshapeIndex) const; + + //! @} + + //! @name Note creation functions + //! @{ + + //! Create a new comment note. + //! Creates a new label under the notes hive and attaches \ref XCAFDoc_NoteComment + //! attribute (derived ftom \ref XCAFDoc_Note). + //! \param [in] theUserName - the user associated with the note. + //! \param [in] theTimeStamp - timestamp of the note. + //! \param [in] theComment - textual comment. + //! \return a handle to the base note attribute. + Standard_EXPORT Handle(XCAFDoc_Note) CreateComment(const TCollection_ExtendedString& theUserName, + const TCollection_ExtendedString& theTimeStamp, + const TCollection_ExtendedString& theComment); + + //! Create a new 'balloon' note. + //! Creates a new label under the notes hive and attaches \ref XCAFDoc_NoteBalloon + //! attribute (derived ftom \ref XCAFDoc_Note). + //! \param [in] theUserName - the user associated with the note. + //! \param [in] theTimeStamp - timestamp of the note. + //! \param [in] theComment - textual comment. + //! \return a handle to the base note attribute. + Standard_EXPORT Handle(XCAFDoc_Note) CreateBalloon(const TCollection_ExtendedString& theUserName, + const TCollection_ExtendedString& theTimeStamp, + const TCollection_ExtendedString& theComment); + + //! Create a new note with data loaded from a binary file. + //! Creates a new label under the notes hive and attaches \ref XCAFDoc_NoteComment + //! attribute (derived ftom \ref XCAFDoc_Note). + //! \param [in] theUserName - the user associated with the note. + //! \param [in] theTimeStamp - timestamp of the note. + //! \param [in] theTitle - file title. + //! \param [in] theMIMEtype - MIME type of the file. + //! \param [in] theFile - input binary file. + //! \return a handle to the base note attribute. + Standard_EXPORT Handle(XCAFDoc_Note) CreateBinData(const TCollection_ExtendedString& theUserName, + const TCollection_ExtendedString& theTimeStamp, + const TCollection_ExtendedString& theTitle, + const TCollection_AsciiString& theMIMEtype, + OSD_File& theFile); + + //! Create a new note with data loaded from a byte data array. + //! Creates a new label under the notes hive and attaches \ref XCAFDoc_NoteComment + //! attribute (derived ftom \ref XCAFDoc_Note). + //! \param [in] theUserName - the user associated with the note. + //! \param [in] theTimeStamp - timestamp of the note. + //! \param [in] theTitle - data title. + //! \param [in] theMIMEtype - MIME type of the file. + //! \param [in] theData - byte data array. + //! \return a handle to the base note attribute. + Standard_EXPORT Handle(XCAFDoc_Note) CreateBinData(const TCollection_ExtendedString& theUserName, + const TCollection_ExtendedString& theTimeStamp, + const TCollection_ExtendedString& theTitle, + const TCollection_AsciiString& theMIMEtype, + const Handle(TColStd_HArray1OfByte)& theData); + + //! @} + + //! @name Get notes from annotated items functions + //! @{ + + //! Gets all note labels of the assembly item. + //! Notes linked to item's subshapes or attributes aren't + //! taken into account. The label sequence isn't cleared beforehand. + //! \param [in] theItemId - assembly item ID. + //! \param [out] theNoteLabels - sequence of labels. + //! \return number of added labels. + Standard_EXPORT Standard_Integer GetNotes(const XCAFDoc_AssemblyItemId& theItemId, + TDF_LabelSequence& theNoteLabels) const; + + //! Gets all note labels of the labeled item. + //! Notes linked to item's attributes aren't + //! taken into account. The label sequence isn't cleared beforehand. + //! \param [in] theItemLabel - item label. + //! \param [out] theNoteLabels - sequence of labels. + //! \return number of added labels. + Standard_EXPORT Standard_Integer GetNotes(const TDF_Label& theItemLabel, + TDF_LabelSequence& theNoteLabels) const; + + //! Gets all note labels of the assembly item's attribute. + //! Notes linked to the item itself or to item's subshapes + //! aren't taken into account. The label sequence isn't cleared beforehand. + //! \param [in] theItemId - assembly item ID. + //! \param [in] theGUID - assembly item's attribute GUID. + //! \param [out] theNoteLabels - sequence of labels. + //! \return number of added labels. + Standard_EXPORT Standard_Integer GetAttrNotes(const XCAFDoc_AssemblyItemId& theItemId, + const Standard_GUID& theGUID, + TDF_LabelSequence& theNoteLabels) const; + + //! Gets all note labels of the labeled item's attribute. + //! Notes linked to the item itself or to item's subshapes + //! aren't taken into account. The label sequence isn't cleared beforehand. + //! \param [in] theItemLabel - item label. + //! \param [in] theGUID - item's attribute GUID. + //! \param [out] theNoteLabels - sequence of labels. + //! \return number of added labels. + Standard_EXPORT Standard_Integer GetAttrNotes(const TDF_Label& theItemLabel, + const Standard_GUID& theGUID, + TDF_LabelSequence& theNoteLabels) const; + + //! Gets all note labels of the annotated item. + //! Notes linked to the item itself or to item's attributes + //! taken into account. The label sequence isn't cleared beforehand. + //! \param [in] theItemId - assembly item ID. + //! \param [in] theSubshapeIndex - assembly item's subshape index. + //! \param [out] theNoteLabels - sequence of labels. + //! \return number of added labels. + Standard_EXPORT Standard_Integer GetSubshapeNotes(const XCAFDoc_AssemblyItemId& theItemId, + Standard_Integer theSubshapeIndex, + TDF_LabelSequence& theNoteLabels) const; + + //! @} + + //! @name Annotation functions + //! @{ + + //! Adds the given note to the assembly item. + //! \param [in] theNoteLabel - note label. + //! \param [in] theItemId - assembly item ID. + //! \return a handle to the assembly reference attribute. + Standard_EXPORT Handle(XCAFDoc_AssemblyItemRef) AddNote(const TDF_Label& theNoteLabel, + const XCAFDoc_AssemblyItemId& theItemId); + + //! Adds the given note to the labeled item. + //! \param [in] theNoteLabel - note label. + //! \param [in] theItemLabel - item label. + //! \return a handle to the assembly reference attribute. + Standard_EXPORT Handle(XCAFDoc_AssemblyItemRef) AddNote(const TDF_Label& theNoteLabel, + const TDF_Label& theItemLabel); + + //! Adds the given note to the assembly item's attribute. + //! \param [in] theNoteLabel - note label. + //! \param [in] theItemId - assembly item ID. + //! \param [in] theGUID - assembly item's attribute GUID. + //! \return a handle to the assembly reference attribute. + Standard_EXPORT Handle(XCAFDoc_AssemblyItemRef) AddNoteToAttr(const TDF_Label& theNoteLabel, + const XCAFDoc_AssemblyItemId& theItemId, + const Standard_GUID& theGUID); + + //! Adds the given note to the labeled item's attribute. + //! \param [in] theNoteLabel - note label. + //! \param [in] theItemLabel - item label. + //! \param [in] theGUID - assembly item's attribute GUID. + //! \return a handle to the assembly reference attribute. + Standard_EXPORT Handle(XCAFDoc_AssemblyItemRef) AddNoteToAttr(const TDF_Label& theNoteLabel, + const TDF_Label& theItemLabel, + const Standard_GUID& theGUID); + + //! Adds the given note to the assembly item's subshape. + //! \param [in] theNoteLabel - note label. + //! \param [in] theItemId - assembly item ID. + //! \param [in] theSubshapeIndex - assembly item's subshape index. + //! \return a handle to the assembly reference attribute. + Standard_EXPORT Handle(XCAFDoc_AssemblyItemRef) AddNoteToSubshape(const TDF_Label& theNoteLabel, + const XCAFDoc_AssemblyItemId& theItemId, + Standard_Integer theSubshapeIndex); + + //! Adds the given note to the labeled item's subshape. + //! \param [in] theNoteLabel - note label. + //! \param [in] theItemLabel - item label. + //! \param [in] theSubshapeIndex - assembly item's subshape index. + //! \return a handle to the assembly reference attribute. + Standard_EXPORT Handle(XCAFDoc_AssemblyItemRef) AddNoteToSubshape(const TDF_Label& theNoteLabel, + const TDF_Label& theItemLabel, + Standard_Integer theSubshapeIndex); + + //! @} + + //! @name Remove annotation functions + //! @{ + + //! Removes the given note from the assembly item. + //! \param [in] theNoteLabel - note label. + //! \param [in] theItemId - assembly item ID. + //! \param [in] theDelIfOrphan - deletes the note from the notes hive + //! if there are no more assembly items + //! linked with the note. + //! \return true if the note is removed, otherwise - false. + Standard_EXPORT Standard_Boolean RemoveNote(const TDF_Label& theNoteLabel, + const XCAFDoc_AssemblyItemId& theItemId, + Standard_Boolean theDelIfOrphan = Standard_False); + + //! Removes the given note from the labeled item. + //! \param [in] theNoteLabel - note label. + //! \param [in] theItemLabel - item label. + //! \param [in] theDelIfOrphan - deletes the note from the notes hive + //! if there are no more labeled items + //! linked with the note. + //! \return true if the note is removed, otherwise - false. + Standard_EXPORT Standard_Boolean RemoveNote(const TDF_Label& theNoteLabel, + const TDF_Label& theItemLabel, + Standard_Boolean theDelIfOrphan = Standard_False); + + //! Removes the given note from the assembly item's subshape. + //! \param [in] theNoteLabel - note label. + //! \param [in] theItemId - assembly item ID. + //! \param [in] theSubshapeIndex - assembly item's subshape index. + //! \param [in] theDelIfOrphan - deletes the note from the notes hive + //! if there are no more assembly item's + //! subshape linked with the note. + //! \return true if the note is removed, otherwise - false. + Standard_EXPORT Standard_Boolean RemoveSubshapeNote(const TDF_Label& theNoteLabel, + const XCAFDoc_AssemblyItemId& theItemId, + Standard_Integer theSubshapeIndex, + Standard_Boolean theDelIfOrphan = Standard_False); + + //! Removes the given note from the labeled item's subshape. + //! \param [in] theNoteLabel - note label. + //! \param [in] theItemLabel - item label. + //! \param [in] theSubshapeIndex - labeled item's subshape index. + //! \param [in] theDelIfOrphan - deletes the note from the notes hive + //! if there are no more assembly item's + //! subshape linked with the note. + //! \return true if the note is removed, otherwise - false. + Standard_EXPORT Standard_Boolean RemoveSubshapeNote(const TDF_Label& theNoteLabel, + const TDF_Label& theItemLabel, + Standard_Integer theSubshapeIndex, + Standard_Boolean theDelIfOrphan = Standard_False); + + //! Removes a note from the assembly item's attribute. + //! \param [in] theNoteLabel - note label. + //! \param [in] theItemId - assembly item ID. + //! \param [in] theGUID - assembly item's attribute GUID. + //! \param [in] theDelIfOrphan - deletes the note from the notes hive + //! if there are no more assembly item's + //! attribute linked with the note. + //! \return true if the note is removed, otherwise - false. + Standard_EXPORT Standard_Boolean RemoveAttrNote(const TDF_Label& theNoteLabel, + const XCAFDoc_AssemblyItemId& theItemId, + const Standard_GUID& theGUID, + Standard_Boolean theDelIfOrphan = Standard_False); + + //! Removes a note from the labeled item's attribute. + //! \param [in] theNoteLabel - note label. + //! \param [in] theItemLabel - item label. + //! \param [in] theGUID - labeled item's attribute GUID. + //! \param [in] theDelIfOrphan - deletes the note from the notes hive + //! if there are no more assembly item's + //! attribute linked with the note. + //! \return true if the note is removed, otherwise - false. + Standard_EXPORT Standard_Boolean RemoveAttrNote(const TDF_Label& theNoteLabel, + const TDF_Label& theItemLabel, + const Standard_GUID& theGUID, + Standard_Boolean theDelIfOrphan = Standard_False); + + //! Removes all notes from the assembly item. + //! \param [in] theItemId - assembly item ID. + //! \param [in] theDelIfOrphan - deletes removed notes from the notes + //! hive if there are no more annotated items + //! linked with the notes. + //! \return true if the notes are removed, otherwise - false. + Standard_EXPORT Standard_Boolean RemoveAllNotes(const XCAFDoc_AssemblyItemId& theItemId, + Standard_Boolean theDelIfOrphan = Standard_False); + + //! Removes all notes from the labeled item. + //! \param [in] theItemLabel - item label. + //! \param [in] theDelIfOrphan - deletes removed notes from the notes + //! hive if there are no more annotated items + //! linked with the notes. + //! \return true if the notes are removed, otherwise - false. + Standard_EXPORT Standard_Boolean RemoveAllNotes(const TDF_Label& theItemLabel, + Standard_Boolean theDelIfOrphan = Standard_False); + + //! Removes all notes from the assembly item's subshape. + //! \param [in] theItemId - assembly item ID. + //! \param [in] theSubshapeIndex - assembly item's subshape index. + //! \param [in] theDelIfOrphan - deletes removed notes from the notes + //! hive if there are no more annotated items + //! linked with the notes. + //! \return true if the notes are removed, otherwise - false. + Standard_EXPORT Standard_Boolean RemoveAllSubshapeNotes(const XCAFDoc_AssemblyItemId& theItemId, + Standard_Integer theSubshapeIndex, + Standard_Boolean theDelIfOrphan = Standard_False); + + //! Removes all notes from the assembly item's attribute. + //! \param [in] theItemId - assembly item ID. + //! \param [in] theGUID - assembly item's attribute GUID. + //! \param [in] theDelIfOrphan - deletes removed notes from the notes + //! hive if there are no more annotated items + //! linked with the notes. + //! \return true if the notes are removed, otherwise - false. + Standard_EXPORT Standard_Boolean RemoveAllAttrNotes(const XCAFDoc_AssemblyItemId& theItemId, + const Standard_GUID& theGUID, + Standard_Boolean theDelIfOrphan = Standard_False); + + //! Removes all notes from the labeled item's attribute. + //! \param [in] theItemLabel - item label. + //! \param [in] theGUID - labeled item's attribute GUID. + //! \param [in] theDelIfOrphan - deletes removed notes from the notes + //! hive if there are no more annotated items + //! linked with the notes. + //! \return true if the notes are removed, otherwise - false. + Standard_EXPORT Standard_Boolean RemoveAllAttrNotes(const TDF_Label& theItemLabel, + const Standard_GUID& theGUID, + Standard_Boolean theDelIfOrphan = Standard_False); + + //! @} + + //! @name Delete note functions + //! @{ + + //! Deletes the given note. + //! Removes all links with items annotated by the note. + //! \param [in] theNoteLabel - note label. + //! \return true if the note is deleted, otherwise - false. + Standard_EXPORT Standard_Boolean DeleteNote(const TDF_Label& theNoteLabel); + + //! Deletes the given notes. + //! Removes all links with items annotated by the notes. + //! \param [in] theNoteLabels - note label sequence. + //! \return number of deleted notes. + Standard_EXPORT Standard_Integer DeleteNotes(TDF_LabelSequence& theNoteLabels); + + //! Deletes all notes. + //! Clears all annotations. + //! \return number of deleted notes. + Standard_EXPORT Standard_Integer DeleteAllNotes(); + + //! @} + + //! @name Orphan annotated items functions + //! @{ + + //! Returns number of notes that aren't linked to annotated items. + Standard_EXPORT Standard_Integer NbOrphanNotes() const; + + //! Returns note labels that aren't linked to annotated items. + //! The label sequence isn't cleared beforehand. + //! \param [out] theNoteLabels - sequence of labels. + Standard_EXPORT void GetOrphanNotes(TDF_LabelSequence& theNoteLabels) const; + + //! Deletes all notes that aren't linked to annotated items. + //! \return number of deleted notes. + Standard_EXPORT Standard_Integer DeleteOrphanNotes(); + + //! @} + +public: + + // Overrides TDF_Attribute virtuals + Standard_EXPORT const Standard_GUID& ID() const Standard_OVERRIDE; + Standard_EXPORT Handle(TDF_Attribute) NewEmpty() const Standard_OVERRIDE; + Standard_EXPORT void Restore(const Handle(TDF_Attribute)& theAttrFrom) Standard_OVERRIDE; + Standard_EXPORT void Paste(const Handle(TDF_Attribute)& theAttrInto, + const Handle(TDF_RelocationTable)& theRT) const Standard_OVERRIDE; + Standard_EXPORT Standard_OStream& Dump(Standard_OStream& theOS) const Standard_OVERRIDE; + +}; + +#endif // _XCAFDoc_NotesTool_HeaderFile diff --git a/src/XCAFDoc/XCAFDoc_PartId.hxx b/src/XCAFDoc/XCAFDoc_PartId.hxx new file mode 100644 index 0000000000..2e71081bf4 --- /dev/null +++ b/src/XCAFDoc/XCAFDoc_PartId.hxx @@ -0,0 +1,22 @@ +// Copyright (c) 1998-1999 Matra Datavision +// Copyright (c) 1999-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _XCAFDoc_PartId_HeaderFile +#define _XCAFDoc_PartId_HeaderFile + +#include + +typedef TCollection_AsciiString XCAFDoc_PartId; + +#endif // _XCAFDoc_PartId_HeaderFile diff --git a/src/XCAFDoc/XCAFDoc_ViewTool.cxx b/src/XCAFDoc/XCAFDoc_ViewTool.cxx index 9bf93ca729..1cc6679457 100644 --- a/src/XCAFDoc/XCAFDoc_ViewTool.cxx +++ b/src/XCAFDoc/XCAFDoc_ViewTool.cxx @@ -117,6 +117,136 @@ TDF_Label XCAFDoc_ViewTool::AddView() return aViewL; } +//======================================================================= +//function : SetView +//purpose : +//======================================================================= +void XCAFDoc_ViewTool::SetView(const TDF_LabelSequence& theShapes, + const TDF_LabelSequence& theGDTs, + const TDF_LabelSequence& theClippingPlanes, + const TDF_LabelSequence& theNotes, + const TDF_LabelSequence& theAnnotations, + const TDF_Label& theViewL) const +{ + if (!IsView(theViewL)) + return; + + Handle(XCAFDoc_GraphNode) aChGNode; + Handle(XCAFDoc_GraphNode) aShapeGNode; + Handle(XCAFDoc_GraphNode) aGDTGNode; + Handle(XCAFDoc_GraphNode) aPlaneGNode; + Handle(XCAFDoc_GraphNode) aNoteGNode; + Handle(XCAFDoc_GraphNode) aAnnotGNode; + + if (theViewL.FindAttribute(XCAFDoc::ViewRefShapeGUID(), aChGNode)) { + while (aChGNode->NbFathers() > 0) { + aShapeGNode = aChGNode->GetFather(1); + aShapeGNode->UnSetChild(aChGNode); + if (aShapeGNode->NbChildren() == 0) + aShapeGNode->ForgetAttribute(XCAFDoc::ViewRefShapeGUID()); + } + theViewL.ForgetAttribute(XCAFDoc::ViewRefShapeGUID()); + } + if (theViewL.FindAttribute(XCAFDoc::ViewRefGDTGUID(), aChGNode)) { + while (aChGNode->NbFathers() > 0) { + aGDTGNode = aChGNode->GetFather(1); + aGDTGNode->UnSetChild(aChGNode); + if (aGDTGNode->NbChildren() == 0) + aGDTGNode->ForgetAttribute(XCAFDoc::ViewRefGDTGUID()); + } + theViewL.ForgetAttribute(XCAFDoc::ViewRefGDTGUID()); + } + if (theViewL.FindAttribute(XCAFDoc::ViewRefPlaneGUID(), aChGNode)) { + while (aChGNode->NbFathers() > 0) { + aPlaneGNode = aChGNode->GetFather(1); + aPlaneGNode->UnSetChild(aChGNode); + if (aPlaneGNode->NbChildren() == 0) + aPlaneGNode->ForgetAttribute(XCAFDoc::ViewRefGDTGUID()); + } + theViewL.ForgetAttribute(XCAFDoc::ViewRefPlaneGUID()); + } + + if (!theViewL.FindAttribute(XCAFDoc::ViewRefShapeGUID(), aChGNode) && theShapes.Length() > 0) { + aChGNode = new XCAFDoc_GraphNode; + aChGNode = XCAFDoc_GraphNode::Set(theViewL); + aChGNode->SetGraphID(XCAFDoc::ViewRefShapeGUID()); + } + for (Standard_Integer i = theShapes.Lower(); i <= theShapes.Upper(); i++) + { + if (!theShapes.Value(i).FindAttribute(XCAFDoc::ViewRefShapeGUID(), aShapeGNode)) { + aShapeGNode = new XCAFDoc_GraphNode; + aShapeGNode = XCAFDoc_GraphNode::Set(theShapes.Value(i)); + } + aShapeGNode->SetGraphID(XCAFDoc::ViewRefShapeGUID()); + aShapeGNode->SetChild(aChGNode); + aChGNode->SetFather(aShapeGNode); + } + + if (!theViewL.FindAttribute(XCAFDoc::ViewRefGDTGUID(), aChGNode) && theGDTs.Length() > 0) { + aChGNode = new XCAFDoc_GraphNode; + aChGNode = XCAFDoc_GraphNode::Set(theViewL); + aChGNode->SetGraphID(XCAFDoc::ViewRefGDTGUID()); + } + for (Standard_Integer i = theGDTs.Lower(); i <= theGDTs.Upper(); i++) + { + if (!theGDTs.Value(i).FindAttribute(XCAFDoc::ViewRefGDTGUID(), aGDTGNode)) { + aGDTGNode = new XCAFDoc_GraphNode; + aGDTGNode = XCAFDoc_GraphNode::Set(theGDTs.Value(i)); + } + aGDTGNode->SetGraphID(XCAFDoc::ViewRefGDTGUID()); + aGDTGNode->SetChild(aChGNode); + aChGNode->SetFather(aGDTGNode); + } + + if (!theViewL.FindAttribute(XCAFDoc::ViewRefPlaneGUID(), aChGNode) && theClippingPlanes.Length() > 0) { + aChGNode = new XCAFDoc_GraphNode; + aChGNode = XCAFDoc_GraphNode::Set(theViewL); + aChGNode->SetGraphID(XCAFDoc::ViewRefPlaneGUID()); + } + for (Standard_Integer i = theClippingPlanes.Lower(); i <= theClippingPlanes.Upper(); i++) + { + if (!theClippingPlanes.Value(i).FindAttribute(XCAFDoc::ViewRefPlaneGUID(), aPlaneGNode)) { + aPlaneGNode = new XCAFDoc_GraphNode; + aPlaneGNode = XCAFDoc_GraphNode::Set(theClippingPlanes.Value(i)); + } + aPlaneGNode->SetGraphID(XCAFDoc::ViewRefPlaneGUID()); + aPlaneGNode->SetChild(aChGNode); + aChGNode->SetFather(aPlaneGNode); + } + + if (!theViewL.FindAttribute(XCAFDoc::ViewRefPlaneGUID(), aChGNode) && theClippingPlanes.Length() > 0) { + aChGNode = new XCAFDoc_GraphNode; + aChGNode = XCAFDoc_GraphNode::Set(theViewL); + aChGNode->SetGraphID(XCAFDoc::ViewRefPlaneGUID()); + } + for (Standard_Integer i = theNotes.Lower(); i <= theNotes.Upper(); i++) + { + if (!theNotes.Value(i).FindAttribute(XCAFDoc::ViewRefNoteGUID(), aNoteGNode)) { + aNoteGNode = new XCAFDoc_GraphNode; + aNoteGNode = XCAFDoc_GraphNode::Set(theNotes.Value(i)); + } + aNoteGNode->SetGraphID(XCAFDoc::ViewRefNoteGUID()); + aNoteGNode->SetChild(aChGNode); + aChGNode->SetFather(aNoteGNode); + } + + if (!theViewL.FindAttribute(XCAFDoc::ViewRefAnnotationGUID(), aChGNode) && theAnnotations.Length() > 0) { + aChGNode = new XCAFDoc_GraphNode; + aChGNode = XCAFDoc_GraphNode::Set(theViewL); + aChGNode->SetGraphID(XCAFDoc::ViewRefAnnotationGUID()); + } + for (Standard_Integer i = theAnnotations.Lower(); i <= theAnnotations.Upper(); i++) + { + if (!theAnnotations.Value(i).FindAttribute(XCAFDoc::ViewRefAnnotationGUID(), aNoteGNode)) { + aAnnotGNode = new XCAFDoc_GraphNode; + aAnnotGNode = XCAFDoc_GraphNode::Set(theNotes.Value(i)); + } + aAnnotGNode->SetGraphID(XCAFDoc::ViewRefAnnotationGUID()); + aAnnotGNode->SetChild(aChGNode); + aChGNode->SetFather(aAnnotGNode); + } +} + //======================================================================= //function : SetView //purpose : @@ -431,6 +561,54 @@ Standard_Boolean XCAFDoc_ViewTool::GetRefClippingPlaneLabel(const TDF_Label& the return Standard_True; } +//======================================================================= +//function : GetRefNoteLabel +//purpose : +//======================================================================= +Standard_Boolean XCAFDoc_ViewTool::GetRefNoteLabel(const TDF_Label& theViewL, + TDF_LabelSequence& theNoteLabels) const +{ + theNoteLabels.Clear(); + Handle(TDataStd_TreeNode) aNode; + if (!theViewL.FindAttribute(XCAFDoc::ViewRefGUID(), aNode) || !aNode->HasFather()) { + Handle(XCAFDoc_GraphNode) aGNode; + if (theViewL.FindAttribute(XCAFDoc::ViewRefNoteGUID(), aGNode) && aGNode->NbFathers() > 0) { + for (Standard_Integer i = 1; i <= aGNode->NbFathers(); i++) + theNoteLabels.Append(aGNode->GetFather(i)->Label()); + return Standard_True; + } + else + return Standard_False; + } + + theNoteLabels.Append(aNode->Father()->Label()); + return Standard_True; +} + +//======================================================================= +//function : GetRefAnnotationLabel +//purpose : +//======================================================================= +Standard_Boolean XCAFDoc_ViewTool::GetRefAnnotationLabel(const TDF_Label& theViewL, + TDF_LabelSequence& theAnnotationLabels) const +{ + theAnnotationLabels.Clear(); + Handle(TDataStd_TreeNode) aNode; + if (!theViewL.FindAttribute(XCAFDoc::ViewRefGUID(), aNode) || !aNode->HasFather()) { + Handle(XCAFDoc_GraphNode) aGNode; + if (theViewL.FindAttribute(XCAFDoc::ViewRefAnnotationGUID(), aGNode) && aGNode->NbFathers() > 0) { + for (Standard_Integer i = 1; i <= aGNode->NbFathers(); i++) + theAnnotationLabels.Append(aGNode->GetFather(i)->Label()); + return Standard_True; + } + else + return Standard_False; + } + + theAnnotationLabels.Append(aNode->Father()->Label()); + return Standard_True; +} + //======================================================================= //function : GetViewLabelsForShape //purpose : @@ -488,6 +666,44 @@ Standard_Boolean XCAFDoc_ViewTool::GetViewLabelsForClippingPlane(const TDF_Label return aResult; } +//======================================================================= +//function : GetViewLabelsForNote +//purpose : +//======================================================================= +Standard_Boolean XCAFDoc_ViewTool::GetViewLabelsForNote(const TDF_Label& theNoteL, + TDF_LabelSequence& theViews) const +{ + Handle(XCAFDoc_GraphNode) aGNode; + Standard_Boolean aResult = Standard_False; + if (theNoteL.FindAttribute(XCAFDoc::ViewRefNoteGUID(), aGNode) && aGNode->NbChildren() > 0) { + for (Standard_Integer i = 1; i <= aGNode->NbChildren(); i++) + { + theViews.Append(aGNode->GetChild(i)->Label()); + } + aResult = Standard_True; + } + return aResult; +} + +//======================================================================= +//function : GetViewLabelsForAnnotation +//purpose : +//======================================================================= +Standard_Boolean XCAFDoc_ViewTool::GetViewLabelsForAnnotation(const TDF_Label& theAnnotationL, + TDF_LabelSequence& theViews) const +{ + Handle(XCAFDoc_GraphNode) aGNode; + Standard_Boolean aResult = Standard_False; + if (theAnnotationL.FindAttribute(XCAFDoc::ViewRefAnnotationGUID(), aGNode) && aGNode->NbChildren() > 0) { + for (Standard_Integer i = 1; i <= aGNode->NbChildren(); i++) + { + theViews.Append(aGNode->GetChild(i)->Label()); + } + aResult = Standard_True; + } + return aResult; +} + //======================================================================= //function : IsLocked //purpose : diff --git a/src/XCAFDoc/XCAFDoc_ViewTool.hxx b/src/XCAFDoc/XCAFDoc_ViewTool.hxx index a995df5255..7fcd681007 100644 --- a/src/XCAFDoc/XCAFDoc_ViewTool.hxx +++ b/src/XCAFDoc/XCAFDoc_ViewTool.hxx @@ -63,7 +63,15 @@ public: //! Returns a sequence of View labels currently stored //! in the View table Standard_EXPORT void GetViewLabels (TDF_LabelSequence& theLabels) const; - + + //! Sets a link with GUID + Standard_EXPORT void SetView(const TDF_LabelSequence& theShapes, + const TDF_LabelSequence& theGDTs, + const TDF_LabelSequence& theClippingPlanes, + const TDF_LabelSequence& theNotes, + const TDF_LabelSequence& theAnnotations, + const TDF_Label& theViewL) const; + //! Sets a link with GUID Standard_EXPORT void SetView (const TDF_LabelSequence& theShapes, const TDF_LabelSequence& theGDTs, @@ -90,7 +98,13 @@ public: //! Returns all View labels defined for label ClippingPlaneL Standard_EXPORT Standard_Boolean GetViewLabelsForClippingPlane(const TDF_Label& theClippingPlaneL, TDF_LabelSequence& theViews) const; - + + //! Returns all View labels defined for label NoteL + Standard_EXPORT Standard_Boolean GetViewLabelsForNote(const TDF_Label& theNoteL, TDF_LabelSequence& theViews) const; + + //! Returns all View labels defined for label AnnotationL + Standard_EXPORT Standard_Boolean GetViewLabelsForAnnotation(const TDF_Label& theAnnotationL, TDF_LabelSequence& theViews) const; + //! Adds a view definition to a View table and returns its label Standard_EXPORT TDF_Label AddView() ; @@ -106,6 +120,14 @@ public: //! Returns False if the theViewL is not in View table Standard_EXPORT Standard_Boolean GetRefClippingPlaneLabel(const TDF_Label& theViewL, TDF_LabelSequence& theClippingPlaneLabels) const; + //! Returns Notes labels defined for label theViewL + //! Returns False if the theViewL is not in View table + Standard_EXPORT Standard_Boolean GetRefNoteLabel(const TDF_Label& theViewL, TDF_LabelSequence& theNoteLabels) const; + + //! Returns Annotation labels defined for label theViewL + //! Returns False if the theViewL is not in View table + Standard_EXPORT Standard_Boolean GetRefAnnotationLabel(const TDF_Label& theViewL, TDF_LabelSequence& theAnnotationLabels) const; + //! Returns true if the given View is marked as locked Standard_EXPORT Standard_Boolean IsLocked(const TDF_Label& theViewL) const; diff --git a/src/XDEDRAW/FILES b/src/XDEDRAW/FILES index 9787e20eac..a896d4c95e 100644 --- a/src/XDEDRAW/FILES +++ b/src/XDEDRAW/FILES @@ -14,3 +14,5 @@ XDEDRAW_Shapes.cxx XDEDRAW_Shapes.hxx XDEDRAW_Views.cxx XDEDRAW_Views.hxx +XDEDRAW_Notes.cxx +XDEDRAW_Notes.hxx diff --git a/src/XDEDRAW/XDEDRAW.cxx b/src/XDEDRAW/XDEDRAW.cxx index c70cbfb3aa..53e003c83b 100644 --- a/src/XDEDRAW/XDEDRAW.cxx +++ b/src/XDEDRAW/XDEDRAW.cxx @@ -87,6 +87,7 @@ #include #include #include +#include #include #include #include @@ -1171,6 +1172,7 @@ void XDEDRAW::Init(Draw_Interpretor& di) XDEDRAW_Props::InitCommands ( di ); XDEDRAW_GDTs::InitCommands ( di ); XDEDRAW_Views::InitCommands(di); + XDEDRAW_Notes::InitCommands(di); XDEDRAW_Common::InitCommands ( di );//moved from EXE } diff --git a/src/XDEDRAW/XDEDRAW_Notes.cxx b/src/XDEDRAW/XDEDRAW_Notes.cxx new file mode 100644 index 0000000000..05fc1e75f6 --- /dev/null +++ b/src/XDEDRAW/XDEDRAW_Notes.cxx @@ -0,0 +1,1438 @@ +// Created on: 2017-08-09 +// Created by: Sergey NIKONOV +// Copyright (c) 2016 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +struct cmd +{ + const char* name; + Standard_Integer nargsreq; + const char* use; +}; + +Draw_Interpretor& operator << (Draw_Interpretor& di, const cmd& c) +{ + di << "Use: " << c.use << "\n"; + return di; +} + +Draw_Interpretor& operator << (Draw_Interpretor& di, const TDF_Label& L) +{ + TCollection_AsciiString anEntry; + TDF_Tool::Entry(L, anEntry); + di << anEntry; + return di; +} + +Draw_Interpretor& operator << (Draw_Interpretor& di, const Handle(XCAFDoc_Note)& note) +{ + TCollection_AsciiString anEntry; + TDF_Tool::Entry(note->Label(), anEntry); + di << anEntry; + return di; +} + +Draw_Interpretor& operator << (Draw_Interpretor& di, const Handle(XCAFDoc_AssemblyItemRef)& ref) +{ + TCollection_AsciiString anEntry; + TDF_Tool::Entry(ref->Label(), anEntry); + di << anEntry; + return di; +} + +//======================================================================= +//function : noteCount +//purpose : returns number of annotated items, notes and orphan notes +//======================================================================= +static const cmd XNoteCount = { + "XNoteCount", 2, "XNoteCount Doc" +}; +static Standard_Integer +noteCount(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteCount; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[1], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + Handle(XCAFDoc_NotesTool) aNotesTool = XCAFDoc_DocumentTool::NotesTool(aDoc->Main()); + + di + << aNotesTool->NbAnnotatedItems() << " " + << aNotesTool->NbNotes() << " " + << aNotesTool->NbOrphanNotes() + ; + return 0; +} + +//======================================================================= +//function : noteNotes +//purpose : returns list of all notes +//======================================================================= +static const cmd XNoteNotes = { + "XNoteNotes", 2, "XNoteNotes Doc" +}; +static Standard_Integer +noteNotes(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteNotes; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[1], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + Handle(XCAFDoc_NotesTool) aNotesTool = XCAFDoc_DocumentTool::NotesTool(aDoc->Main()); + + TDF_LabelSequence aNotes; + aNotesTool->GetNotes(aNotes); + for (TDF_LabelSequence::Iterator anIt(aNotes); anIt.More(); anIt.Next()) + { + Handle(XCAFDoc_Note) aNote = XCAFDoc_Note::Get(anIt.Value()); + di << aNote << " "; + } + + return 0; +} + +//======================================================================= +//function : noteAnnotations +//purpose : returns list of all notes +//======================================================================= +static const cmd XNoteAnnotations = { + "XNoteAnnotations", 2, "XNoteAnnotations Doc" +}; +static Standard_Integer +noteAnnotations(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteAnnotations; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[1], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + Handle(XCAFDoc_NotesTool) aNotesTool = XCAFDoc_DocumentTool::NotesTool(aDoc->Main()); + + TDF_LabelSequence aAnnotations; + aNotesTool->GetAnnotatedItems(aAnnotations); + for (TDF_LabelSequence::Iterator anIt(aAnnotations); anIt.More(); anIt.Next()) + { + Handle(XCAFDoc_AssemblyItemRef) aRef = XCAFDoc_AssemblyItemRef::Get(anIt.Value()); + di << aRef << " "; + } + + return 0; +} + +//======================================================================= +//function : noteCreateBalloon +//purpose : creates a 'balloon' note and returns a new note entry +//======================================================================= +static const cmd XNoteCreateBalloon = { + "XNoteCreateBalloon", 3, "XNoteCreateBalloon Doc comment [--user name] [--time stamp]" +}; +static Standard_Integer +noteCreateBalloon(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteCreateBalloon; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Standard_Integer iarg = 0; + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[++iarg], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + TCollection_ExtendedString aComment = argv[++iarg]; + TCollection_ExtendedString aUsername, aTimestamp; + + for (++iarg; iarg < argc; ++iarg) + { + TCollection_AsciiString opt = argv[iarg]; + if (opt == "--user") + { + if (++iarg >= argc) + { + di << "Error: user name is expected.\n" << myCommand; + return 1; + } + aUsername = argv[iarg]; + } + else if (opt == "--time") + { + if (++iarg >= argc) + { + di << "Error: timestamp is expected.\n" << myCommand; + return 1; + } + aTimestamp = argv[iarg]; + } + } + + Handle(XCAFDoc_NotesTool) aNotesTool = XCAFDoc_DocumentTool::NotesTool(aDoc->Main()); + Handle(XCAFDoc_Note) aNote = aNotesTool->CreateBalloon(aUsername, aTimestamp, aComment); + if (!aNote) + { + di << "Error: couldn't create a comment note.\n"; + return 1; + } + + di << aNote; + return 0; +} + +//======================================================================= +//function : noteCreateComment +//purpose : creates a comment note and returns a new note entry +//======================================================================= +static const cmd XNoteCreateComment = { + "XNoteCreateComment", 3, "XNoteCreateComment Doc comment [--user name] [--time stamp]" +}; +static Standard_Integer +noteCreateComment(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteCreateComment; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Standard_Integer iarg = 0; + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[++iarg], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + TCollection_ExtendedString aComment = argv[++iarg]; + TCollection_ExtendedString aUsername, aTimestamp; + + for (++iarg; iarg < argc; ++iarg) + { + TCollection_AsciiString opt = argv[iarg]; + if (opt == "--user") + { + if (++iarg >= argc) + { + di << "Error: user name is expected.\n" << myCommand; + return 1; + } + aUsername = argv[iarg]; + } + else if (opt == "--time") + { + if (++iarg >= argc) + { + di << "Error: timestamp is expected.\n" << myCommand; + return 1; + } + aTimestamp = argv[iarg]; + } + } + + Handle(XCAFDoc_NotesTool) aNotesTool = XCAFDoc_DocumentTool::NotesTool(aDoc->Main()); + Handle(XCAFDoc_Note) aNote = aNotesTool->CreateComment(aUsername, aTimestamp, aComment); + if (!aNote) + { + di << "Error: couldn't create a comment note.\n"; + return 1; + } + + di << aNote; + return 0; +} + +//======================================================================= +//function : noteCreateBinData +//purpose : creates a binary data note and returns a new note entry +//======================================================================= +static const cmd XNoteCreateBinData = { + "XNoteCreateBinData", 5, "XNoteCreateBinData Doc title <--file path | --data data> [--mime type] [--user name] [--time stamp]" +}; +static Standard_Integer +noteCreateBinData(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteCreateBinData; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Standard_Integer iarg = 0; + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[++iarg], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + Standard_Boolean aFromFile = Standard_False; + Standard_Boolean aFromData = Standard_False; + TCollection_ExtendedString aFilename; + Handle(TColStd_HArray1OfByte) aData; + TCollection_ExtendedString aTitle = argv[++iarg]; + TCollection_AsciiString aMIMEtype; + TCollection_ExtendedString aUsername, aTimestamp; + + for (++iarg; iarg < argc; ++iarg) + { + TCollection_AsciiString opt = argv[iarg]; + if (opt == "--file") + { + if (aFromData) + { + di << "Error: data can be taken either from a file or a data array.\n" << myCommand; + return 1; + } + if (++iarg >= argc) + { + di << "Error: file path is expected.\n" << myCommand; + return 1; + } + aFilename = argv[iarg]; + aFromFile = Standard_True; + } + else if (opt == "--data") + { + if (aFromFile) + { + di << "Error: data can be taken either from a file or a data array.\n" << myCommand; + return 1; + } + if (++iarg >= argc) + { + di << "Error: data array is expected.\n" << myCommand; + return 1; + } + Standard_SStream ss(std::ios_base::in | std::ios_base::out | std::ios_base::binary); + ss << argv[iarg]; + Standard_Integer len = static_cast(ss.tellp()); + aData = new TColStd_HArray1OfByte(1, len); + for (Standard_Integer i = 1; i <= len && !ss.eof(); ++i) + { + ss >> aData->ChangeValue(i); + } + aFromData = Standard_True; + } + else if (opt == "--mime") + { + if (++iarg >= argc) + { + di << "Error: MIME type is expected.\n" << myCommand; + return 1; + } + aMIMEtype = argv[iarg]; + } + else if (opt == "--user") + { + if (++iarg >= argc) + { + di << "Error: user name is expected.\n" << myCommand; + return 1; + } + aUsername = argv[iarg]; + } + else if (opt == "--time") + { + if (++iarg >= argc) + { + di << "Error: timestamp is expected.\n" << myCommand; + return 1; + } + aTimestamp = argv[iarg]; + } + } + + Handle(XCAFDoc_NotesTool) aNotesTool = XCAFDoc_DocumentTool::NotesTool(aDoc->Main()); + + Handle(XCAFDoc_Note) aNote; + if (aFromFile) + { + OSD_Path aPath(aFilename); + OSD_File aFile(aPath); + aFile.Open(OSD_ReadOnly, OSD_Protection()); + aNote = aNotesTool->CreateBinData(aUsername, aTimestamp, aTitle, aMIMEtype, aFile); + } + else if (aFromData) + { + aNote = aNotesTool->CreateBinData(aUsername, aTimestamp, aTitle, aMIMEtype, aData); + } + else + { + di << "Error: data can be taken either from a file or a data array.\n" << myCommand; + return 1; + } + + if (!aNote) + { + di << "Error: couldn't create a binary data note.\n"; + return 1; + } + + di << aNote; + return 0; +} + +//======================================================================= +//function : noteDelete +//purpose : deletes a note by the entry +//======================================================================= +static const cmd XNoteDelete = { + "XNoteDelete", 3, "XNoteDelete Doc note" +}; +static Standard_Integer +noteDelete(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteDelete; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[1], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + TCollection_ExtendedString anEntry = argv[2]; + + TDF_Label aLabel; + TDF_Tool::Label(aDoc->GetData(), anEntry, aLabel); + if (aLabel.IsNull()) + { + di << anEntry << ": invalid note entry.\n"; + return 1; + } + + Handle(XCAFDoc_NotesTool) aNotesTool = XCAFDoc_DocumentTool::NotesTool(aDoc->Main()); + if (!aNotesTool->DeleteNote(aLabel)) + { + di << "Error: couldn't delete note.\n"; + return 1; + } + + return 0; +} + +//======================================================================= +//function : noteDeleteAll +//purpose : deletes all notes +//======================================================================= +static const cmd XNoteDeleteAll = { + "XNoteDeleteAll", 2, "XNoteDeleteAll Doc" +}; +static Standard_Integer +noteDeleteAll(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteDeleteAll; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[1], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + Handle(XCAFDoc_NotesTool) aNotesTool = XCAFDoc_DocumentTool::NotesTool(aDoc->Main()); + Standard_Integer nbDeleted = aNotesTool->DeleteAllNotes(); + + di << nbDeleted; + return 0; +} + +//======================================================================= +//function : noteDeleteOrphan +//purpose : deletes all orphan notes +//======================================================================= +static const cmd XNoteDeleteOrphan = { + "XNoteDeleteOrphan", 2, "XNoteDeleteOrphan Doc" +}; +static Standard_Integer +noteDeleteOrphan(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteDeleteOrphan; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[1], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + Handle(XCAFDoc_NotesTool) aNotesTool = XCAFDoc_DocumentTool::NotesTool(aDoc->Main()); + Standard_Integer nbDeleted = aNotesTool->DeleteOrphanNotes(); + + di << nbDeleted; + return 0; +} + +//======================================================================= +//function : noteAdd +//purpose : adds a note to a labeled item +//======================================================================= +static const cmd XNoteAdd = { + "XNoteAdd", 4, "XNoteAdd Doc note item [--attr guid | --subshape num]" +}; +static Standard_Integer +noteAdd(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteAdd; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Standard_Integer iarg = 0; + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[++iarg], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + TCollection_AsciiString aNoteEntry = argv[++iarg]; + TCollection_AsciiString anItemEntry = argv[++iarg]; + + TDF_Label aNoteLabel; + TDF_Tool::Label(aDoc->GetData(), aNoteEntry, aNoteLabel); + if (aNoteLabel.IsNull()) + { + di << aNoteEntry << ": invalid note entry.\n"; + return 1; + } + XCAFDoc_AssemblyItemId anItemId(anItemEntry); + TDF_Label anItemLabel; + TDF_Tool::Label(aDoc->GetData(), anItemId.GetPath().Last(), anItemLabel); + if (anItemLabel.IsNull()) + { + di << anItemId.ToString() << ": invalid item id.\n"; + return 1; + } + + Standard_GUID aGUID; + Standard_Integer aSubshape = 0; + Standard_Boolean aHasGUID = Standard_False; + Standard_Boolean aHasSubshape = Standard_False; + for (++iarg; iarg < argc; ++iarg) + { + TCollection_AsciiString opt = argv[iarg]; + if (opt == "--attr") + { + if (aHasSubshape) + { + di << "Error: either attribute guid or subshape index must be specified.\n" << myCommand; + return 1; + } + if (++iarg >= argc) + { + di << "Error: attribute guid is expected.\n" << myCommand; + return 1; + } + TCollection_AsciiString arg = argv[iarg]; + aGUID = arg.ToCString(); + aHasGUID = Standard_True; + } + else if (opt == "--subshape") + { + if (++iarg >= argc) + { + di << "Error: subshape index is expected.\n" << myCommand; + return 1; + } + TCollection_AsciiString arg = argv[iarg]; + if (!arg.IsIntegerValue()) + { + di << "Error: subshape index must be integer value.\n" << myCommand; + return 1; + } + aSubshape = arg.IntegerValue(); + aHasSubshape = Standard_True; + } + } + + Handle(XCAFDoc_NotesTool) aNotesTool = XCAFDoc_DocumentTool::NotesTool(aDoc->Main()); + + Handle(XCAFDoc_AssemblyItemRef) aRef; + if (aHasGUID) + { + aRef = aNotesTool->AddNoteToAttr(aNoteLabel, anItemId, aGUID); + } + else if (aHasSubshape) + { + aRef = aNotesTool->AddNoteToSubshape(aNoteLabel, anItemId, aSubshape); + } + else + { + aRef = aNotesTool->AddNote(aNoteLabel, anItemId); + } + + if (!aRef) + { + di << "Error: couldn't add note.\n"; + return 1; + } + + if (aRef->IsOrphan()) + { + di << "Error: annotated item is invalid\n"; + return 1; + } + + di << aRef; + return 0; +} + +//======================================================================= +//function : noteRemove +//purpose : removes a note from a labeled item +//======================================================================= +static const cmd XNoteRemove = { + "XNoteRemove", 4, "XNoteRemove Doc item note [--attr guid | --subshape num] [--del-orphan]" +}; +static Standard_Integer +noteRemove(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteRemove; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Standard_Integer iarg = 0; + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[++iarg], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + TCollection_AsciiString anItemEntry = argv[++iarg]; + TCollection_AsciiString aNoteEntry = argv[++iarg]; + + TDF_Label aNoteLabel; + TDF_Tool::Label(aDoc->GetData(), aNoteEntry, aNoteLabel); + if (aNoteLabel.IsNull()) + { + di << aNoteEntry << ": invalid note entry.\n"; + return 1; + } + XCAFDoc_AssemblyItemId anItemId(anItemEntry); + TDF_Label anItemLabel; + TDF_Tool::Label(aDoc->GetData(), anItemId.GetPath().Last(), anItemLabel); + if (anItemLabel.IsNull()) + { + di << anItemId.ToString() << ": invalid item id.\n"; + return 1; + } + + Standard_GUID aGUID; + Standard_Integer aSubshape = 0; + Standard_Boolean aHasGUID = Standard_False; + Standard_Boolean aHasSubshape = Standard_False; + Standard_Boolean aDelOrphan = Standard_False; + for (++iarg; iarg < argc; ++iarg) + { + TCollection_AsciiString opt = argv[iarg]; + if (opt == "--attr") + { + if (aHasSubshape) + { + di << "Error: either attribute guid or subshape index must be specified.\n" << myCommand; + return 1; + } + if (++iarg >= argc) + { + di << "Error: attribute guid is expected.\n" << myCommand; + return 1; + } + TCollection_AsciiString arg = argv[iarg]; + aGUID = arg.ToCString(); + aHasGUID = Standard_True; + } + else if (opt == "--subshape") + { + if (aHasGUID) + { + di << "Error: either attribute guid or subshape index must be specified.\n" << myCommand; + return 1; + } + if (++iarg >= argc) + { + di << "Error: subshape index is expected.\n" << myCommand; + return 1; + } + TCollection_AsciiString arg = argv[iarg]; + if (!arg.IsIntegerValue()) + { + di << "Error: subshape index must be integer value.\n" << myCommand; + return 1; + } + aSubshape = arg.IntegerValue(); + aHasSubshape = Standard_True; + } + else if (opt == "--del-orphan") + { + aDelOrphan = Standard_True; + } + } + + Handle(XCAFDoc_NotesTool) aNotesTool = XCAFDoc_DocumentTool::NotesTool(aDoc->Main()); + + if (aHasGUID) + { + di << aNotesTool->RemoveAttrNote(aNoteLabel, anItemId, aGUID, aDelOrphan); + } + else if (aHasSubshape) + { + di << aNotesTool->RemoveSubshapeNote(aNoteLabel, anItemId, aSubshape, aDelOrphan); + } + else + { + di << aNotesTool->RemoveNote(aNoteLabel, anItemId, aDelOrphan); + } + + return 0; +} + +//======================================================================= +//function : noteRemoveAll +//purpose : removes all notes from a labeled item +//======================================================================= +static const cmd XNoteRemoveAll = { + "XNoteRemoveAll", 3, "XNoteRemoveAll Doc item [--attr guid] [--subshape num] [--del-orphan]" +}; +static Standard_Integer +noteRemoveAll(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteRemoveAll; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Standard_Integer iarg = 0; + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[++iarg], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + TCollection_AsciiString anItemEntry = argv[++iarg]; + + XCAFDoc_AssemblyItemId anItemId(anItemEntry); + TDF_Label anItemLabel; + TDF_Tool::Label(aDoc->GetData(), anItemId.GetPath().Last(), anItemLabel); + if (anItemLabel.IsNull()) + { + di << anItemId.ToString() << ": invalid item id.\n"; + return 1; + } + + Standard_GUID aGUID; + Standard_Integer aSubshape = 0; + Standard_Boolean aHasGUID = Standard_False; + Standard_Boolean aHasSubshape = Standard_False; + Standard_Boolean aDelOrphan = Standard_False; + for (++iarg; iarg < argc; ++iarg) + { + TCollection_AsciiString opt = argv[iarg]; + if (opt == "--attr") + { + if (aHasSubshape) + { + di << "Error: either attribute guid or subshape index must be specified.\n" << myCommand; + return 1; + } + if (++iarg >= argc) + { + di << "Error: attribute guid is expected.\n" << myCommand; + return 1; + } + TCollection_AsciiString arg = argv[iarg]; + aGUID = arg.ToCString(); + aHasGUID = Standard_True; + } + else if (opt == "--subshape") + { + if (aHasGUID) + { + di << "Error: either attribute guid or subshape index must be specified.\n" << myCommand; + return 1; + } + if (++iarg >= argc) + { + di << "Error: subshape index is expected.\n" << myCommand; + return 1; + } + TCollection_AsciiString arg = argv[iarg]; + if (!arg.IsIntegerValue()) + { + di << "Error: subshape index must be integer value.\n" << myCommand; + return 1; + } + aSubshape = arg.IntegerValue(); + aHasSubshape = Standard_True; + } + else if (opt == "--del-orphan") + { + aDelOrphan = Standard_True; + } + } + + Handle(XCAFDoc_NotesTool) aNotesTool = XCAFDoc_DocumentTool::NotesTool(aDoc->Main()); + + if (aHasGUID) + { + di << aNotesTool->RemoveAllAttrNotes(anItemId, aGUID, aDelOrphan); + } + else if (aHasSubshape) + { + di << aNotesTool->RemoveAllSubshapeNotes(anItemId, aSubshape, aDelOrphan); + } + else + { + di << aNotesTool->RemoveAllNotes(anItemId, aDelOrphan); + } + + return 0; +} + +//======================================================================= +//function : noteFindAnnotated +//purpose : find a label of the annotated item +//======================================================================= +static const cmd XNoteFindAnnotated = { + "XNoteFindAnnotated", 3, "XNoteFindAnnotated Doc item [--attr guid | --subshape num]" +}; +static Standard_Integer +noteFindAnnotated(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteFindAnnotated; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Standard_Integer iarg = 0; + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[++iarg], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + TCollection_AsciiString anItemEntry = argv[++iarg]; + + XCAFDoc_AssemblyItemId anItemId(anItemEntry); + TDF_Label anItemLabel; + TDF_Tool::Label(aDoc->GetData(), anItemId.GetPath().Last(), anItemLabel); + if (anItemLabel.IsNull()) + { + di << anItemId.ToString() << ": invalid item id.\n"; + return 1; + } + + Standard_GUID aGUID; + Standard_Integer aSubshape = 0; + Standard_Boolean aHasGUID = Standard_False; + Standard_Boolean aHasSubshape = Standard_False; + for (++iarg; iarg < argc; ++iarg) + { + TCollection_AsciiString opt = argv[iarg]; + if (opt == "--attr") + { + if (aHasSubshape) + { + di << "Error: either attribute guid or subshape index must be specified.\n" << myCommand; + return 1; + } + if (++iarg >= argc) + { + di << "Error: attribute guid is expected.\n" << myCommand; + return 1; + } + TCollection_AsciiString arg = argv[iarg]; + aGUID = arg.ToCString(); + aHasGUID = Standard_True; + } + else if (opt == "--subshape") + { + if (++iarg >= argc) + { + di << "Error: subshape index is expected.\n" << myCommand; + return 1; + } + TCollection_AsciiString arg = argv[iarg]; + if (!arg.IsIntegerValue()) + { + di << "Error: subshape index must be integer value.\n" << myCommand; + return 1; + } + aSubshape = arg.IntegerValue(); + aHasSubshape = Standard_True; + } + } + + Handle(XCAFDoc_NotesTool) aNotesTool = XCAFDoc_DocumentTool::NotesTool(aDoc->Main()); + + if (aHasGUID) + { + di << aNotesTool->FindAnnotatedItemAttr(anItemId, aGUID); + } + else if (aHasSubshape) + { + di << aNotesTool->FindAnnotatedItemSubshape(anItemId, aSubshape); + } + else + { + di << aNotesTool->FindAnnotatedItem(anItemId); + } + + return 0; +} + +//======================================================================= +//function : noteGetNotes +//purpose : get notes of the labeled item +//======================================================================= +static const cmd XNoteGetNotes = { + "XNoteGetNotes", 3, "XNoteGetNotes Doc item [--attr guid | --subshape num]" +}; +static Standard_Integer +noteGetNotes(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteGetNotes; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Standard_Integer iarg = 0; + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[++iarg], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + TCollection_AsciiString anItemEntry = argv[++iarg]; + + XCAFDoc_AssemblyItemId anItemId(anItemEntry); + TDF_Label anItemLabel; + TDF_Tool::Label(aDoc->GetData(), anItemId.GetPath().Last(), anItemLabel); + if (anItemLabel.IsNull()) + { + di << anItemId.ToString() << ": invalid item id.\n"; + return 1; + } + + Standard_GUID aGUID; + Standard_Integer aSubshape = 0; + Standard_Boolean aHasGUID = Standard_False; + Standard_Boolean aHasSubshape = Standard_False; + for (++iarg; iarg < argc; ++iarg) + { + TCollection_AsciiString opt = argv[iarg]; + if (opt == "--attr") + { + if (aHasSubshape) + { + di << "Error: either attribute guid or subshape index must be specified.\n" << myCommand; + return 1; + } + if (++iarg >= argc) + { + di << "Error: attribute guid is expected.\n" << myCommand; + return 1; + } + TCollection_AsciiString arg = argv[iarg]; + aGUID = arg.ToCString(); + aHasGUID = Standard_True; + } + else if (opt == "--subshape") + { + if (++iarg >= argc) + { + di << "Error: subshape index is expected.\n" << myCommand; + return 1; + } + TCollection_AsciiString arg = argv[iarg]; + if (!arg.IsIntegerValue()) + { + di << "Error: subshape index must be integer value.\n" << myCommand; + return 1; + } + aSubshape = arg.IntegerValue(); + aHasSubshape = Standard_True; + } + } + + Handle(XCAFDoc_NotesTool) aNotesTool = XCAFDoc_DocumentTool::NotesTool(aDoc->Main()); + + TDF_LabelSequence aNotes; + if (aHasGUID) + { + aNotesTool->GetAttrNotes(anItemId, aGUID, aNotes); + } + else if (aHasSubshape) + { + aNotesTool->GetSubshapeNotes(anItemId, aSubshape, aNotes); + } + else + { + aNotesTool->GetNotes(anItemId, aNotes); + } + + for (TDF_LabelSequence::Iterator anIt(aNotes); anIt.More(); anIt.Next()) + { + di << anIt.Value() << " "; + } + + return 0; +} + +//======================================================================= +//function : noteUsername +//purpose : gets a note username +//======================================================================= +static const cmd XNoteUsername = { + "XNoteUsername", 3, "XNoteUsername Doc note" +}; +static Standard_Integer +noteUsername(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteUsername; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Standard_Integer iarg = 0; + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[++iarg], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + TCollection_AsciiString aNoteEntry = argv[++iarg]; + + TDF_Label aNoteLabel; + TDF_Tool::Label(aDoc->GetData(), aNoteEntry, aNoteLabel); + if (aNoteLabel.IsNull()) + { + di << aNoteEntry << ": invalid note entry.\n"; + return 1; + } + + Handle(XCAFDoc_Note) aNote = XCAFDoc_Note::Get(aNoteLabel); + if (aNote.IsNull()) + { + di << aNoteEntry << ": not a note entry.\n"; + return 1; + } + + di << aNote->UserName(); + + return 0; +} + +//======================================================================= +//function : noteTimestamp +//purpose : gets a note timestamp +//======================================================================= +static const cmd XNoteTimestamp = { + "XNoteTimestamp", 3, "XNoteTimestamp Doc note" +}; +static Standard_Integer +noteTimestamp(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteTimestamp; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Standard_Integer iarg = 0; + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[++iarg], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + TCollection_AsciiString aNoteEntry = argv[++iarg]; + + TDF_Label aNoteLabel; + TDF_Tool::Label(aDoc->GetData(), aNoteEntry, aNoteLabel); + if (aNoteLabel.IsNull()) + { + di << aNoteEntry << ": invalid note entry.\n"; + return 1; + } + + Handle(XCAFDoc_Note) aNote = XCAFDoc_Note::Get(aNoteLabel); + if (aNote.IsNull()) + { + di << aNoteEntry << ": not a note entry.\n"; + return 1; + } + + di << aNote->TimeStamp(); + + return 0; +} + +//======================================================================= +//function : noteDump +//purpose : dump a note contents +//======================================================================= +static const cmd XNoteDump = { + "XNoteDump", 3, "XNoteDump Doc note" +}; +static Standard_Integer +noteDump(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteDump; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Standard_Integer iarg = 0; + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[++iarg], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + TCollection_AsciiString aNoteEntry = argv[++iarg]; + + TDF_Label aNoteLabel; + TDF_Tool::Label(aDoc->GetData(), aNoteEntry, aNoteLabel); + if (aNoteLabel.IsNull()) + { + di << aNoteEntry << ": invalid note entry.\n"; + return 1; + } + + Handle(XCAFDoc_Note) aNote = XCAFDoc_Note::Get(aNoteLabel); + if (aNote.IsNull()) + { + di << aNoteEntry << ": not a note entry.\n"; + return 1; + } + + di << "Username : " << aNote->UserName() << "\n"; + di << "Timestamp : " << aNote->TimeStamp() << "\n"; + di << "Type : " << aNote->get_type_name() << "\n"; + if (Handle(XCAFDoc_NoteComment) aComment = Handle(XCAFDoc_NoteComment)::DownCast(aNote)) + { + di << "Comment : " << aComment->Comment() << "\n"; + } + else if (Handle(XCAFDoc_NoteBalloon) aBalloon = Handle(XCAFDoc_NoteBalloon)::DownCast(aNote)) + { + di << "Comment : " << aBalloon->Comment() << "\n"; + } + else if (Handle(XCAFDoc_NoteBinData) aBinData = Handle(XCAFDoc_NoteBinData)::DownCast(aNote)) + { + di << "Title : " << aBinData->Title() << "\n"; + di << "MIME type : " << aBinData->MIMEtype() << "\n"; + di << "Size : " << aBinData->Size() << "\n"; + static Standard_Integer theMaxLen = 64; + const Handle(TColStd_HArray1OfByte)& aData = aBinData->Data(); + if (!aData.IsNull()) + { + di << "Data : "; + Standard_Integer aLen = Min(aData->Length(), theMaxLen); + for (Standard_Integer i = 1; i <= aLen; ++i) + { + Standard_SStream ss; ss << aData->Value(i); + di << ss.str().c_str(); + } + if (aData->Length() > theMaxLen) + di << " ..."; + di << "\n"; + } + } + + return 0; +} + +//======================================================================= +//function : noteIsRefOrphan +//purpose : checks if a ref is orphan +//======================================================================= +static const cmd XNoteRefDump = { + "XNoteRefDump", 3, "XNoteRefDump Doc ref" +}; +static Standard_Integer +noteRefDump(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteRefDump; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Standard_Integer iarg = 0; + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[++iarg], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + TCollection_AsciiString aRefEntry = argv[++iarg]; + + TDF_Label aRefLabel; + TDF_Tool::Label(aDoc->GetData(), aRefEntry, aRefLabel); + if (aRefLabel.IsNull()) + { + di << aRefEntry << ": invalid reference entry.\n"; + return 1; + } + + Handle(XCAFDoc_AssemblyItemRef) aRef = XCAFDoc_AssemblyItemRef::Get(aRefLabel); + if (aRef.IsNull()) + { + di << aRefEntry << ": not a reference entry.\n"; + return 1; + } + + di << "Item : " << aRef->GetItem().ToString() << "\n"; + if (aRef->HasExtraRef()) + { + if (aRef->IsGUID()) + { + Standard_SStream ss; + aRef->GetGUID().ShallowDump(ss); + di << "Attribute : " << ss.str().c_str() << "\n"; + } + else if (aRef->IsSubshapeIndex()) + di << "Subshape : " << aRef->GetSubshapeIndex() << "\n"; + } + di << "Orphan : " << (aRef->IsOrphan() ? "yes" : "no") << "\n"; + + return 0; +} + +//======================================================================= +//function : noteIsRefOrphan +//purpose : checks if a ref is orphan +//======================================================================= +static const cmd XNoteIsRefOrphan = { + "XNoteIsRefOrphan", 3, "XNoteIsRefOrphan Doc ref" +}; +static Standard_Integer +noteIsRefOrphan(Draw_Interpretor& di, Standard_Integer argc, const char** argv) +{ + static const cmd& myCommand = XNoteIsRefOrphan; + + if (argc < myCommand.nargsreq) + { + di << myCommand; + return 1; + } + + Standard_Integer iarg = 0; + + Handle(TDocStd_Document) aDoc; + DDocStd::GetDocument(argv[++iarg], aDoc); + if (aDoc.IsNull()) + { + return 1; + } + + TCollection_AsciiString aRefEntry = argv[++iarg]; + + TDF_Label aRefLabel; + TDF_Tool::Label(aDoc->GetData(), aRefEntry, aRefLabel); + if (aRefLabel.IsNull()) + { + di << aRefEntry << ": invalid reference entry.\n"; + return 1; + } + + Handle(XCAFDoc_AssemblyItemRef) aRef = XCAFDoc_AssemblyItemRef::Get(aRefLabel); + if (aRef.IsNull()) + { + di << aRefEntry << ": not a reference entry.\n"; + return 1; + } + + di << aRef->IsOrphan(); + + return 0; +} + +//======================================================================= +//function : InitCommands +//purpose : +//======================================================================= +void XDEDRAW_Notes::InitCommands(Draw_Interpretor& di) +{ + static Standard_Boolean initialized = Standard_False; + if (initialized) + { + return; + } + initialized = Standard_True; + + Standard_CString g = "XDE Notes commands"; + + di.Add(XNoteCount.name, XNoteCount.use, + __FILE__, noteCount, g); + di.Add(XNoteNotes.name, XNoteNotes.use, + __FILE__, noteNotes, g); + di.Add(XNoteAnnotations.name, XNoteAnnotations.use, + __FILE__, noteAnnotations, g); + + di.Add(XNoteCreateBalloon.name, XNoteCreateBalloon.use, + __FILE__, noteCreateBalloon, g); + di.Add(XNoteCreateComment.name, XNoteCreateComment.use, + __FILE__, noteCreateComment, g); + di.Add(XNoteCreateBinData.name, XNoteCreateBinData.use, + __FILE__, noteCreateBinData, g); + di.Add(XNoteDelete.name, XNoteDelete.use, + __FILE__, noteDelete, g); + di.Add(XNoteDeleteAll.name, XNoteDeleteAll.use, + __FILE__, noteDeleteAll, g); + di.Add(XNoteDeleteOrphan.name, XNoteDeleteOrphan.use, + __FILE__, noteDeleteOrphan, g); + + di.Add(XNoteAdd.name, XNoteAdd.use, + __FILE__, noteAdd, g); + di.Add(XNoteRemove.name, XNoteRemove.use, + __FILE__, noteRemove, g); + di.Add(XNoteRemoveAll.name, XNoteRemoveAll.use, + __FILE__, noteRemoveAll, g); + + di.Add(XNoteFindAnnotated.name, XNoteFindAnnotated.use, + __FILE__, noteFindAnnotated, g); + di.Add(XNoteGetNotes.name, XNoteGetNotes.use, + __FILE__, noteGetNotes, g); + + di.Add(XNoteUsername.name, XNoteUsername.use, + __FILE__, noteUsername, g); + di.Add(XNoteTimestamp.name, XNoteTimestamp.use, + __FILE__, noteTimestamp, g); + di.Add(XNoteDump.name, XNoteDump.use, + __FILE__, noteDump, g); + + di.Add(XNoteRefDump.name, XNoteRefDump.use, + __FILE__, noteRefDump, g); + di.Add(XNoteIsRefOrphan.name, XNoteIsRefOrphan.use, + __FILE__, noteIsRefOrphan, g); +} diff --git a/src/XDEDRAW/XDEDRAW_Notes.hxx b/src/XDEDRAW/XDEDRAW_Notes.hxx new file mode 100644 index 0000000000..c7392af4ea --- /dev/null +++ b/src/XDEDRAW/XDEDRAW_Notes.hxx @@ -0,0 +1,35 @@ +// Created on: 2017-08-09 +// Created by: Sergey NIKONOV +// Copyright (c) 2016 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _XDEDRAW_Notes_HeaderFile +#define _XDEDRAW_Notes_HeaderFile + +#include +#include +#include + +#include + +//! Contains commands to work with notes +class XDEDRAW_Notes +{ +public: + + DEFINE_STANDARD_ALLOC + + Standard_EXPORT static void InitCommands (Draw_Interpretor& theCommands); +}; + +#endif // _XDEDRAW_Notes_HeaderFile diff --git a/src/XmlMXCAFDoc/FILES b/src/XmlMXCAFDoc/FILES index d93ba3b27c..965d5fbf55 100644 --- a/src/XmlMXCAFDoc/FILES +++ b/src/XmlMXCAFDoc/FILES @@ -1,5 +1,7 @@ XmlMXCAFDoc.cxx XmlMXCAFDoc.hxx +XmlMXCAFDoc_AssemblyItemRefDriver.cxx +XmlMXCAFDoc_AssemblyItemRefDriver.hxx XmlMXCAFDoc_AreaDriver.cxx XmlMXCAFDoc_AreaDriver.hxx XmlMXCAFDoc_CentroidDriver.cxx @@ -29,6 +31,16 @@ XmlMXCAFDoc_MaterialDriver.cxx XmlMXCAFDoc_MaterialDriver.hxx XmlMXCAFDoc_MaterialToolDriver.cxx XmlMXCAFDoc_MaterialToolDriver.hxx +XmlMXCAFDoc_NoteDriver.cxx +XmlMXCAFDoc_NoteDriver.hxx +XmlMXCAFDoc_NoteBalloonDriver.cxx +XmlMXCAFDoc_NoteBalloonDriver.hxx +XmlMXCAFDoc_NoteCommentDriver.cxx +XmlMXCAFDoc_NoteCommentDriver.hxx +XmlMXCAFDoc_NoteBinDataDriver.cxx +XmlMXCAFDoc_NoteBinDataDriver.hxx +XmlMXCAFDoc_NotesToolDriver.cxx +XmlMXCAFDoc_NotesToolDriver.hxx XmlMXCAFDoc_ShapeToolDriver.cxx XmlMXCAFDoc_ShapeToolDriver.hxx XmlMXCAFDoc_ViewToolDriver.cxx diff --git a/src/XmlMXCAFDoc/XmlMXCAFDoc.cxx b/src/XmlMXCAFDoc/XmlMXCAFDoc.cxx index fe596313b0..2ce8743dae 100644 --- a/src/XmlMXCAFDoc/XmlMXCAFDoc.cxx +++ b/src/XmlMXCAFDoc/XmlMXCAFDoc.cxx @@ -19,6 +19,7 @@ #include #include #include +#include #include #include #include @@ -32,6 +33,10 @@ #include #include #include +#include +#include +#include +#include #include #include #include @@ -61,11 +66,15 @@ void XmlMXCAFDoc::AddDrivers (const Handle(XmlMDF_ADriverTable)& aDriverTable, aLocationDriver->SetSharedLocations( &(aNamedShapeDriver->GetShapesLocations()) ); } - aDriverTable -> AddDriver( aLocationDriver); - aDriverTable -> AddDriver (new XmlMXCAFDoc_VolumeDriver (anMsgDrv)); - aDriverTable -> AddDriver (new XmlMXCAFDoc_DatumDriver (anMsgDrv)); - aDriverTable -> AddDriver (new XmlMXCAFDoc_DimTolDriver (anMsgDrv)); - aDriverTable -> AddDriver (new XmlMXCAFDoc_MaterialDriver (anMsgDrv)); + aDriverTable -> AddDriver (aLocationDriver); + aDriverTable -> AddDriver (new XmlMXCAFDoc_AssemblyItemRefDriver(anMsgDrv)); + aDriverTable -> AddDriver (new XmlMXCAFDoc_VolumeDriver (anMsgDrv)); + aDriverTable -> AddDriver (new XmlMXCAFDoc_DatumDriver (anMsgDrv)); + aDriverTable -> AddDriver (new XmlMXCAFDoc_DimTolDriver (anMsgDrv)); + aDriverTable -> AddDriver (new XmlMXCAFDoc_MaterialDriver (anMsgDrv)); + aDriverTable -> AddDriver (new XmlMXCAFDoc_NoteBalloonDriver(anMsgDrv)); + aDriverTable -> AddDriver (new XmlMXCAFDoc_NoteCommentDriver(anMsgDrv)); + aDriverTable -> AddDriver (new XmlMXCAFDoc_NoteBinDataDriver(anMsgDrv)); aDriverTable -> AddDriver (new XmlMXCAFDoc_ColorToolDriver (anMsgDrv)); aDriverTable -> AddDriver (new XmlMXCAFDoc_DocumentToolDriver (anMsgDrv)); @@ -73,6 +82,7 @@ void XmlMXCAFDoc::AddDrivers (const Handle(XmlMDF_ADriverTable)& aDriverTable, aDriverTable -> AddDriver (new XmlMXCAFDoc_ShapeToolDriver (anMsgDrv)); aDriverTable -> AddDriver (new XmlMXCAFDoc_DimTolToolDriver (anMsgDrv)); aDriverTable -> AddDriver (new XmlMXCAFDoc_MaterialToolDriver (anMsgDrv)); + aDriverTable -> AddDriver (new XmlMXCAFDoc_NotesToolDriver (anMsgDrv)); aDriverTable -> AddDriver (new XmlMXCAFDoc_ViewToolDriver (anMsgDrv)); aDriverTable -> AddDriver (new XmlMXCAFDoc_ClippingPlaneToolDriver(anMsgDrv)); } diff --git a/src/XmlMXCAFDoc/XmlMXCAFDoc.hxx b/src/XmlMXCAFDoc/XmlMXCAFDoc.hxx index b82e95b358..9f5582e68e 100644 --- a/src/XmlMXCAFDoc/XmlMXCAFDoc.hxx +++ b/src/XmlMXCAFDoc/XmlMXCAFDoc.hxx @@ -31,6 +31,10 @@ class XmlMXCAFDoc_VolumeDriver; class XmlMXCAFDoc_DatumDriver; class XmlMXCAFDoc_DimTolDriver; class XmlMXCAFDoc_MaterialDriver; +class XmlMXCAFDoc_NotesToolDriver; +class XmlMXCAFDoc_NoteDriver; +class XmlMXCAFDoc_NoteCommentDriver; +class XmlMXCAFDoc_NoteBinDataDriver; class XmlMXCAFDoc_ClippingPlaneToolDriver; class XmlMXCAFDoc_ColorToolDriver; class XmlMXCAFDoc_DocumentToolDriver; diff --git a/src/XmlMXCAFDoc/XmlMXCAFDoc_AssemblyItemRefDriver.cxx b/src/XmlMXCAFDoc/XmlMXCAFDoc_AssemblyItemRefDriver.cxx new file mode 100644 index 0000000000..f9fe9fb022 --- /dev/null +++ b/src/XmlMXCAFDoc/XmlMXCAFDoc_AssemblyItemRefDriver.cxx @@ -0,0 +1,117 @@ +// Created on: 2017-02-14 +// Created by: Sergey NIKONOV +// Copyright (c) 2008-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include +#include +#include +#include + +IMPLEMENT_STANDARD_RTTIEXT(XmlMXCAFDoc_AssemblyItemRefDriver, XmlMDF_ADriver) +IMPLEMENT_DOMSTRING(Path, "path") +IMPLEMENT_DOMSTRING(AttrGUID, "guid") +IMPLEMENT_DOMSTRING(SubshapeIndex, "subshape_index") + +//======================================================================= +//function : +//purpose : +//======================================================================= +XmlMXCAFDoc_AssemblyItemRefDriver::XmlMXCAFDoc_AssemblyItemRefDriver(const Handle(CDM_MessageDriver)& theMsgDriver) +: XmlMDF_ADriver(theMsgDriver, STANDARD_TYPE(XCAFDoc_AssemblyItemRef)->Name()) +{ +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Handle(TDF_Attribute) XmlMXCAFDoc_AssemblyItemRefDriver::NewEmpty() const +{ + return new XCAFDoc_AssemblyItemRef(); +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Standard_Boolean XmlMXCAFDoc_AssemblyItemRefDriver::Paste(const XmlObjMgt_Persistent& theSource, + const Handle(TDF_Attribute)& theTarget, + XmlObjMgt_RRelocationTable& /*theRelocTable*/) const +{ + const XmlObjMgt_Element& anElement = theSource; + + XmlObjMgt_DOMString aPath = anElement.getAttribute(::Path()); + if (aPath == NULL) + return Standard_False; + + Handle(XCAFDoc_AssemblyItemRef) aThis = Handle(XCAFDoc_AssemblyItemRef)::DownCast(theTarget); + if (aThis.IsNull()) + return Standard_False; + + aThis->SetItem(aPath.GetString()); + + XmlObjMgt_DOMString anAttrGUID = anElement.getAttribute(::AttrGUID()); + if (anAttrGUID != NULL) + { + Standard_GUID aGUID(anAttrGUID.GetString()); + aThis->SetGUID(aGUID); + return Standard_True; + } + + XmlObjMgt_DOMString aSubshapeIndex = anElement.getAttribute(::SubshapeIndex()); + if (aSubshapeIndex != NULL) + { + Standard_Integer anIndex; + if (!aSubshapeIndex.GetInteger(anIndex)) + return Standard_False; + + aThis->SetSubshapeIndex(anIndex); + return Standard_True; + } + + return Standard_True; +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +void XmlMXCAFDoc_AssemblyItemRefDriver::Paste(const Handle(TDF_Attribute)& theSource, + XmlObjMgt_Persistent& theTarget, + XmlObjMgt_SRelocationTable& /*theRelocTable*/) const +{ + Handle(XCAFDoc_AssemblyItemRef) aThis = Handle(XCAFDoc_AssemblyItemRef)::DownCast(theSource); + + XmlObjMgt_DOMString aPath(aThis->GetItem().ToString().ToCString()); + theTarget.Element().setAttribute(::Path(), aPath); + + if (aThis->IsGUID()) + { + Standard_GUID aGUID = aThis->GetGUID(); + Standard_Character aGUIDStr[Standard_GUID_SIZE + 1]; + aGUID.ToCString(aGUIDStr); + aGUIDStr[Standard_GUID_SIZE] = '\0'; + XmlObjMgt_DOMString anAttrGUID(aGUIDStr); + theTarget.Element().setAttribute(::AttrGUID(), anAttrGUID); + } + else if (aThis->IsSubshapeIndex()) + { + TCollection_AsciiString aSubshapeIndexStr(aThis->GetSubshapeIndex()); + XmlObjMgt_DOMString aSubshapeIndex(aSubshapeIndexStr.ToCString()); + theTarget.Element().setAttribute(::SubshapeIndex(), aSubshapeIndex); + } + +} diff --git a/src/XmlMXCAFDoc/XmlMXCAFDoc_AssemblyItemRefDriver.hxx b/src/XmlMXCAFDoc/XmlMXCAFDoc_AssemblyItemRefDriver.hxx new file mode 100644 index 0000000000..2be1a1e751 --- /dev/null +++ b/src/XmlMXCAFDoc/XmlMXCAFDoc_AssemblyItemRefDriver.hxx @@ -0,0 +1,55 @@ +// Created on: 2017-02-16 +// Created by: Sergey NIKONOV +// Copyright (c) 2008-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _XmlMXCAFDoc_AssemblyItemRefDriver_HeaderFile +#define _XmlMXCAFDoc_AssemblyItemRefDriver_HeaderFile + +#include +#include + +#include +#include +#include +#include + +class CDM_MessageDriver; +class TDF_Attribute; +class XmlObjMgt_Persistent; + +class XmlMXCAFDoc_AssemblyItemRefDriver; +DEFINE_STANDARD_HANDLE(XmlMXCAFDoc_AssemblyItemRefDriver, XmlMDF_ADriver) + +//! Attribute Driver. +class XmlMXCAFDoc_AssemblyItemRefDriver : public XmlMDF_ADriver +{ +public: + + Standard_EXPORT XmlMXCAFDoc_AssemblyItemRefDriver(const Handle(CDM_MessageDriver)& theMessageDriver); + + Standard_EXPORT Handle(TDF_Attribute) NewEmpty() const Standard_OVERRIDE; + + Standard_EXPORT Standard_Boolean Paste(const XmlObjMgt_Persistent& theSource, + const Handle(TDF_Attribute)& theTarget, + XmlObjMgt_RRelocationTable& theRelocTable) const Standard_OVERRIDE; + + Standard_EXPORT void Paste(const Handle(TDF_Attribute)& theSource, + XmlObjMgt_Persistent& theTarget, + XmlObjMgt_SRelocationTable& theRelocTable) const Standard_OVERRIDE; + + DEFINE_STANDARD_RTTIEXT(XmlMXCAFDoc_AssemblyItemRefDriver, XmlMDF_ADriver) + +}; + +#endif // _XmlMXCAFDoc_AssemblyItemRefDriver_HeaderFile diff --git a/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteBalloonDriver.cxx b/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteBalloonDriver.cxx new file mode 100644 index 0000000000..f3e5449134 --- /dev/null +++ b/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteBalloonDriver.cxx @@ -0,0 +1,52 @@ +// Created on: 2017-08-10 +// Created by: Sergey NIKONOV +// Copyright (c) 2008-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include +#include +#include +#include + +IMPLEMENT_STANDARD_RTTIEXT(XmlMXCAFDoc_NoteBalloonDriver, XmlMXCAFDoc_NoteCommentDriver) + +//======================================================================= +//function : +//purpose : +//======================================================================= +XmlMXCAFDoc_NoteBalloonDriver::XmlMXCAFDoc_NoteBalloonDriver(const Handle(CDM_MessageDriver)& theMsgDriver) + : XmlMXCAFDoc_NoteCommentDriver(theMsgDriver, STANDARD_TYPE(XCAFDoc_NoteBalloon)->Name()) +{ +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Handle(TDF_Attribute) XmlMXCAFDoc_NoteBalloonDriver::NewEmpty() const +{ + return new XCAFDoc_NoteBalloon(); +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +XmlMXCAFDoc_NoteBalloonDriver::XmlMXCAFDoc_NoteBalloonDriver(const Handle(CDM_MessageDriver)& theMsgDriver, + Standard_CString theName) + : XmlMXCAFDoc_NoteCommentDriver(theMsgDriver, theName) +{ + +} diff --git a/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteBalloonDriver.hxx b/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteBalloonDriver.hxx new file mode 100644 index 0000000000..9db95bc4e5 --- /dev/null +++ b/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteBalloonDriver.hxx @@ -0,0 +1,42 @@ +// Created on: 2017-08-10 +// Created by: Sergey NIKONOV +// Copyright (c) 2008-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _XmlMXCAFDoc_NoteBalloonDriver_HeaderFile +#define _XmlMXCAFDoc_NoteBalloonDriver_HeaderFile + +#include + +class XmlMXCAFDoc_NoteBalloonDriver; +DEFINE_STANDARD_HANDLE(XmlMXCAFDoc_NoteBalloonDriver, XmlMXCAFDoc_NoteCommentDriver) + +//! Attribute Driver. +class XmlMXCAFDoc_NoteBalloonDriver : public XmlMXCAFDoc_NoteCommentDriver +{ +public: + + Standard_EXPORT XmlMXCAFDoc_NoteBalloonDriver(const Handle(CDM_MessageDriver)& theMessageDriver); + + Standard_EXPORT Handle(TDF_Attribute) NewEmpty() const Standard_OVERRIDE; + + DEFINE_STANDARD_RTTIEXT(XmlMXCAFDoc_NoteBalloonDriver, XmlMXCAFDoc_NoteCommentDriver) + +protected: + + XmlMXCAFDoc_NoteBalloonDriver(const Handle(CDM_MessageDriver)& theMsgDriver, + Standard_CString theName); + +}; + +#endif // _XmlMXCAFDoc_NoteBalloonDriver_HeaderFile diff --git a/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteBinDataDriver.cxx b/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteBinDataDriver.cxx new file mode 100644 index 0000000000..7a0c9c413e --- /dev/null +++ b/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteBinDataDriver.cxx @@ -0,0 +1,121 @@ +// Created on: 2017-02-14 +// Created by: Sergey NIKONOV +// Copyright (c) 2008-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include +#include +#include +#include +#include +#include + +IMPLEMENT_STANDARD_RTTIEXT(XmlMXCAFDoc_NoteBinDataDriver, XmlMXCAFDoc_NoteDriver) +IMPLEMENT_DOMSTRING(Title, "title") +IMPLEMENT_DOMSTRING(MIMEtype, "mime_type") +IMPLEMENT_DOMSTRING(Size, "size") + +//======================================================================= +//function : +//purpose : +//======================================================================= +XmlMXCAFDoc_NoteBinDataDriver::XmlMXCAFDoc_NoteBinDataDriver(const Handle(CDM_MessageDriver)& theMsgDriver) + : XmlMXCAFDoc_NoteDriver(theMsgDriver, STANDARD_TYPE(XCAFDoc_NoteBinData)->Name()) +{ +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Handle(TDF_Attribute) XmlMXCAFDoc_NoteBinDataDriver::NewEmpty() const +{ + return new XCAFDoc_NoteBinData(); +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Standard_Boolean XmlMXCAFDoc_NoteBinDataDriver::Paste(const XmlObjMgt_Persistent& theSource, + const Handle(TDF_Attribute)& theTarget, + XmlObjMgt_RRelocationTable& theRelocTable) const +{ + XmlMXCAFDoc_NoteDriver::Paste(theSource, theTarget, theRelocTable); + + const XmlObjMgt_Element& anElement = theSource; + + XmlObjMgt_DOMString aTitle = anElement.getAttribute(::Title()); + XmlObjMgt_DOMString aMIMEtype = anElement.getAttribute(::MIMEtype()); + XmlObjMgt_DOMString aSize = anElement.getAttribute(::Size()); + if (aTitle == NULL || aMIMEtype == NULL || aSize == NULL) + return Standard_False; + + Handle(XCAFDoc_NoteBinData) aNote = Handle(XCAFDoc_NoteBinData)::DownCast(theTarget); + if (aNote.IsNull()) + return Standard_False; + + Standard_Integer nbSize = 0; + if (!aSize.GetInteger(nbSize)) + return Standard_False; + + XmlObjMgt_DOMString aDataStr = XmlObjMgt::GetStringValue(theSource); + Standard_SStream anSS(aDataStr.GetString()); + + Handle(TColStd_HArray1OfByte) aData = new TColStd_HArray1OfByte(1, nbSize); + for (Standard_Integer i = 1; i <= nbSize; ++i) + { + Standard_Byte aValue; + anSS >> aValue; + aData->ChangeValue(i) = aValue; + } + + aNote->Set(aTitle.GetString(), aMIMEtype.GetString(), aData); + + return Standard_True; +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +void XmlMXCAFDoc_NoteBinDataDriver::Paste(const Handle(TDF_Attribute)& theSource, + XmlObjMgt_Persistent& theTarget, + XmlObjMgt_SRelocationTable& theRelocTable) const +{ + XmlMXCAFDoc_NoteDriver::Paste(theSource, theTarget, theRelocTable); + + Handle(XCAFDoc_NoteBinData) aNote = Handle(XCAFDoc_NoteBinData)::DownCast(theSource); + + XmlObjMgt_DOMString aTitle(TCollection_AsciiString(aNote->Title()).ToCString()); + XmlObjMgt_DOMString aMIMEtype(aNote->MIMEtype().ToCString()); + + theTarget.Element().setAttribute(::Title(), aTitle); + theTarget.Element().setAttribute(::MIMEtype(), aMIMEtype); + theTarget.Element().setAttribute(::Size(), aNote->Size()); + + if (aNote->Size() > 0) + { + const Handle(TColStd_HArray1OfByte)& aData = aNote->Data(); + LDOM_OSStream anOSS(aNote->Size()); + for (Standard_Integer i = aData->Lower(); i <= aData->Upper(); ++i) + { + anOSS << std::hex << aData->Value(i); + } + Standard_Character* dump = (Standard_Character*)anOSS.str(); // copying! Don't forget to delete it. + XmlObjMgt::SetStringValue(theTarget, dump, Standard_True); + delete[] dump; + } +} diff --git a/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteBinDataDriver.hxx b/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteBinDataDriver.hxx new file mode 100644 index 0000000000..cca07af0bb --- /dev/null +++ b/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteBinDataDriver.hxx @@ -0,0 +1,45 @@ +// Created on: 2017-02-14 +// Created by: Sergey NIKONOV +// Copyright (c) 2008-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _XmlMXCAFDoc_NoteBinDataDriver_HeaderFile +#define _XmlMXCAFDoc_NoteBinDataDriver_HeaderFile + +#include + +class XmlMXCAFDoc_NoteBinDataDriver; +DEFINE_STANDARD_HANDLE(XmlMXCAFDoc_NoteBinDataDriver, XmlMXCAFDoc_NoteDriver) + +//! Attribute Driver. +class XmlMXCAFDoc_NoteBinDataDriver : public XmlMXCAFDoc_NoteDriver +{ +public: + + Standard_EXPORT XmlMXCAFDoc_NoteBinDataDriver(const Handle(CDM_MessageDriver)& theMessageDriver); + + Standard_EXPORT Handle(TDF_Attribute) NewEmpty() const Standard_OVERRIDE; + + Standard_EXPORT Standard_Boolean Paste(const XmlObjMgt_Persistent& theSource, + const Handle(TDF_Attribute)& theTarget, + XmlObjMgt_RRelocationTable& theRelocTable) const Standard_OVERRIDE; + + Standard_EXPORT void Paste(const Handle(TDF_Attribute)& theSource, + XmlObjMgt_Persistent& theTarget, + XmlObjMgt_SRelocationTable& theRelocTable) const Standard_OVERRIDE; + + DEFINE_STANDARD_RTTIEXT(XmlMXCAFDoc_NoteBinDataDriver, XmlMXCAFDoc_NoteDriver) + +}; + +#endif // _XmlMXCAFDoc_NoteBinDataDriver_HeaderFile diff --git a/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteCommentDriver.cxx b/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteCommentDriver.cxx new file mode 100644 index 0000000000..b3e661e4e0 --- /dev/null +++ b/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteCommentDriver.cxx @@ -0,0 +1,95 @@ +// Created on: 2017-02-14 +// Created by: Sergey NIKONOV +// Copyright (c) 2008-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include +#include +#include +#include + +IMPLEMENT_STANDARD_RTTIEXT(XmlMXCAFDoc_NoteCommentDriver, XmlMXCAFDoc_NoteDriver) +IMPLEMENT_DOMSTRING(Comment, "comment") + +//======================================================================= +//function : +//purpose : +//======================================================================= +XmlMXCAFDoc_NoteCommentDriver::XmlMXCAFDoc_NoteCommentDriver(const Handle(CDM_MessageDriver)& theMsgDriver) + : XmlMXCAFDoc_NoteDriver(theMsgDriver, STANDARD_TYPE(XCAFDoc_NoteComment)->Name()) +{ +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Handle(TDF_Attribute) XmlMXCAFDoc_NoteCommentDriver::NewEmpty() const +{ + return new XCAFDoc_NoteComment(); +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Standard_Boolean XmlMXCAFDoc_NoteCommentDriver::Paste(const XmlObjMgt_Persistent& theSource, + const Handle(TDF_Attribute)& theTarget, + XmlObjMgt_RRelocationTable& theRelocTable) const +{ + XmlMXCAFDoc_NoteDriver::Paste(theSource, theTarget, theRelocTable); + + const XmlObjMgt_Element& anElement = theSource; + + XmlObjMgt_DOMString aComment = anElement.getAttribute(::Comment()); + if (aComment == NULL) + return Standard_False; + + Handle(XCAFDoc_NoteComment) aNote = Handle(XCAFDoc_NoteComment)::DownCast(theTarget); + if (aNote.IsNull()) + return Standard_False; + + aNote->Set(aComment.GetString()); + + return Standard_True; +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +void XmlMXCAFDoc_NoteCommentDriver::Paste(const Handle(TDF_Attribute)& theSource, + XmlObjMgt_Persistent& theTarget, + XmlObjMgt_SRelocationTable& theRelocTable) const +{ + XmlMXCAFDoc_NoteDriver::Paste(theSource, theTarget, theRelocTable); + + Handle(XCAFDoc_NoteComment) aNote = Handle(XCAFDoc_NoteComment)::DownCast(theSource); + + XmlObjMgt_DOMString aComment(TCollection_AsciiString(aNote->TimeStamp()).ToCString()); + + theTarget.Element().setAttribute(::Comment(), aComment); +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +XmlMXCAFDoc_NoteCommentDriver::XmlMXCAFDoc_NoteCommentDriver(const Handle(CDM_MessageDriver)& theMsgDriver, + Standard_CString theName) + : XmlMXCAFDoc_NoteDriver(theMsgDriver, theName) +{ + +} diff --git a/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteCommentDriver.hxx b/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteCommentDriver.hxx new file mode 100644 index 0000000000..dba66c6542 --- /dev/null +++ b/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteCommentDriver.hxx @@ -0,0 +1,50 @@ +// Created on: 2017-02-14 +// Created by: Sergey NIKONOV +// Copyright (c) 2008-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _XmlMXCAFDoc_NoteCommentDriver_HeaderFile +#define _XmlMXCAFDoc_NoteCommentDriver_HeaderFile + +#include + +class XmlMXCAFDoc_NoteCommentDriver; +DEFINE_STANDARD_HANDLE(XmlMXCAFDoc_NoteCommentDriver, XmlMXCAFDoc_NoteDriver) + +//! Attribute Driver. +class XmlMXCAFDoc_NoteCommentDriver : public XmlMXCAFDoc_NoteDriver +{ +public: + + Standard_EXPORT XmlMXCAFDoc_NoteCommentDriver(const Handle(CDM_MessageDriver)& theMessageDriver); + + Standard_EXPORT Handle(TDF_Attribute) NewEmpty() const Standard_OVERRIDE; + + Standard_EXPORT Standard_Boolean Paste(const XmlObjMgt_Persistent& theSource, + const Handle(TDF_Attribute)& theTarget, + XmlObjMgt_RRelocationTable& theRelocTable) const Standard_OVERRIDE; + + Standard_EXPORT void Paste(const Handle(TDF_Attribute)& theSource, + XmlObjMgt_Persistent& theTarget, + XmlObjMgt_SRelocationTable& theRelocTable) const Standard_OVERRIDE; + + DEFINE_STANDARD_RTTIEXT(XmlMXCAFDoc_NoteCommentDriver, XmlMXCAFDoc_NoteDriver) + +protected: + + XmlMXCAFDoc_NoteCommentDriver(const Handle(CDM_MessageDriver)& theMsgDriver, + Standard_CString theName); + +}; + +#endif // _XmlMXCAFDoc_NoteCommentDriver_HeaderFile diff --git a/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteDriver.cxx b/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteDriver.cxx new file mode 100644 index 0000000000..63b2b8d18b --- /dev/null +++ b/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteDriver.cxx @@ -0,0 +1,78 @@ +// Created on: 2017-02-14 +// Created by: Sergey NIKONOV +// Copyright (c) 2008-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include +#include +#include +#include + +IMPLEMENT_STANDARD_RTTIEXT(XmlMXCAFDoc_NoteDriver, XmlMDF_ADriver) +IMPLEMENT_DOMSTRING(UserName, "user_name") +IMPLEMENT_DOMSTRING(TimeStamp, "time_stamp") + +//======================================================================= +//function : +//purpose : +//======================================================================= +XmlMXCAFDoc_NoteDriver::XmlMXCAFDoc_NoteDriver(const Handle(CDM_MessageDriver)& theMsgDriver, + Standard_CString theName) + : XmlMDF_ADriver(theMsgDriver, theName) +{ +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Standard_Boolean XmlMXCAFDoc_NoteDriver::Paste(const XmlObjMgt_Persistent& theSource, + const Handle(TDF_Attribute)& theTarget, + XmlObjMgt_RRelocationTable& /*theRelocTable*/) const +{ + const XmlObjMgt_Element& anElement = theSource; + + XmlObjMgt_DOMString aUserName = anElement.getAttribute(::UserName()); + XmlObjMgt_DOMString aTimeStamp = anElement.getAttribute(::TimeStamp()); + if (aUserName == NULL || aTimeStamp == NULL) + return Standard_False; + + Handle(XCAFDoc_Note) aNote = Handle(XCAFDoc_Note)::DownCast(theTarget); + if (aNote.IsNull()) + return Standard_False; + + aNote->Set(aUserName.GetString(), aTimeStamp.GetString()); + + return Standard_True; +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +void XmlMXCAFDoc_NoteDriver::Paste(const Handle(TDF_Attribute)& theSource, + XmlObjMgt_Persistent& theTarget, + XmlObjMgt_SRelocationTable& /*theRelocTable*/) const +{ + Handle(XCAFDoc_Note) aNote = Handle(XCAFDoc_Note)::DownCast(theSource); + if (aNote.IsNull()) + return; + + XmlObjMgt_DOMString aUserName(TCollection_AsciiString(aNote->UserName()).ToCString()); + XmlObjMgt_DOMString aTimeStamp(TCollection_AsciiString(aNote->TimeStamp()).ToCString()); + + theTarget.Element().setAttribute(::UserName(), aUserName); + theTarget.Element().setAttribute(::TimeStamp(), aTimeStamp); +} diff --git a/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteDriver.hxx b/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteDriver.hxx new file mode 100644 index 0000000000..9dca3f7f48 --- /dev/null +++ b/src/XmlMXCAFDoc/XmlMXCAFDoc_NoteDriver.hxx @@ -0,0 +1,56 @@ +// Created on: 2017-02-14 +// Created by: Sergey NIKONOV +// Copyright (c) 2008-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _XmlMXCAFDoc_NoteDriver_HeaderFile +#define _XmlMXCAFDoc_NoteDriver_HeaderFile + +#include +#include + +#include +#include +#include +#include + +class CDM_MessageDriver; +class TDF_Attribute; +class XmlObjMgt_Persistent; + +class XmlMXCAFDoc_NoteDriver; +DEFINE_STANDARD_HANDLE(XmlMXCAFDoc_NoteDriver, XmlMDF_ADriver) + +//! Attribute Driver. +class XmlMXCAFDoc_NoteDriver : public XmlMDF_ADriver +{ +public: + + Standard_EXPORT Standard_Boolean Paste(const XmlObjMgt_Persistent& theSource, + const Handle(TDF_Attribute)& theTarget, + XmlObjMgt_RRelocationTable& theRelocTable) const Standard_OVERRIDE; + + Standard_EXPORT void Paste(const Handle(TDF_Attribute)& theSource, + XmlObjMgt_Persistent& theTarget, + XmlObjMgt_SRelocationTable& theRelocTable) const Standard_OVERRIDE; + + DEFINE_STANDARD_RTTIEXT(XmlMXCAFDoc_NoteDriver, XmlMDF_ADriver) + +protected: + + XmlMXCAFDoc_NoteDriver(const Handle(CDM_MessageDriver)& theMsgDriver, + Standard_CString theName); + +}; + +#endif // _XmlMXCAFDoc_NoteDriver_HeaderFile diff --git a/src/XmlMXCAFDoc/XmlMXCAFDoc_NotesToolDriver.cxx b/src/XmlMXCAFDoc/XmlMXCAFDoc_NotesToolDriver.cxx new file mode 100644 index 0000000000..dd946f5034 --- /dev/null +++ b/src/XmlMXCAFDoc/XmlMXCAFDoc_NotesToolDriver.cxx @@ -0,0 +1,62 @@ +// Created on: 2017-02-14 +// Created by: Sergey NIKONOV +// Copyright (c) 2008-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include +#include +#include +#include + +IMPLEMENT_STANDARD_RTTIEXT(XmlMXCAFDoc_NotesToolDriver, XmlMDF_ADriver) + +//======================================================================= +//function : +//purpose : +//======================================================================= +XmlMXCAFDoc_NotesToolDriver::XmlMXCAFDoc_NotesToolDriver(const Handle(CDM_MessageDriver)& theMsgDriver) + : XmlMDF_ADriver(theMsgDriver, STANDARD_TYPE(XCAFDoc_NotesTool)->Name()) +{ +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Handle(TDF_Attribute) XmlMXCAFDoc_NotesToolDriver::NewEmpty() const +{ + return new XCAFDoc_NotesTool(); +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +Standard_Boolean XmlMXCAFDoc_NotesToolDriver::Paste(const XmlObjMgt_Persistent& /*theSource*/, + const Handle(TDF_Attribute)& /*theTarget*/, + XmlObjMgt_RRelocationTable& /*theRelocTable*/) const +{ + return Standard_True; +} + +//======================================================================= +//function : +//purpose : +//======================================================================= +void XmlMXCAFDoc_NotesToolDriver::Paste(const Handle(TDF_Attribute)& /*theSource*/, + XmlObjMgt_Persistent& /*theTarget*/, + XmlObjMgt_SRelocationTable& /*theRelocTable*/) const +{ +} diff --git a/src/XmlMXCAFDoc/XmlMXCAFDoc_NotesToolDriver.hxx b/src/XmlMXCAFDoc/XmlMXCAFDoc_NotesToolDriver.hxx new file mode 100644 index 0000000000..6de16cbe54 --- /dev/null +++ b/src/XmlMXCAFDoc/XmlMXCAFDoc_NotesToolDriver.hxx @@ -0,0 +1,55 @@ +// Created on: 2017-02-14 +// Created by: Sergey NIKONOV +// Copyright (c) 2008-2017 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _XmlMXCAFDoc_NotesToolDriver_HeaderFile +#define _XmlMXCAFDoc_NotesToolDriver_HeaderFile + +#include +#include + +#include +#include +#include +#include + +class CDM_MessageDriver; +class TDF_Attribute; +class XmlObjMgt_Persistent; + +class XmlMXCAFDoc_NotesToolDriver; +DEFINE_STANDARD_HANDLE(XmlMXCAFDoc_NotesToolDriver, XmlMDF_ADriver) + +//! Attribute Driver. +class XmlMXCAFDoc_NotesToolDriver : public XmlMDF_ADriver +{ +public: + + Standard_EXPORT XmlMXCAFDoc_NotesToolDriver(const Handle(CDM_MessageDriver)& theMsgDriver); + + Standard_EXPORT Handle(TDF_Attribute) NewEmpty() const Standard_OVERRIDE; + + Standard_EXPORT Standard_Boolean Paste(const XmlObjMgt_Persistent& theSource, + const Handle(TDF_Attribute)& theTarget, + XmlObjMgt_RRelocationTable& theRelocTable) const Standard_OVERRIDE; + + Standard_EXPORT void Paste(const Handle(TDF_Attribute)& theSource, + XmlObjMgt_Persistent& theTarget, + XmlObjMgt_SRelocationTable& theRelocTable) const Standard_OVERRIDE; + + DEFINE_STANDARD_RTTIEXT(XmlMXCAFDoc_NotesToolDriver, XmlMDF_ADriver) + +}; + +#endif // _XmlMXCAFDoc_NotesToolDriver_HeaderFile diff --git a/tests/gdt/grids.list b/tests/gdt/grids.list index 4f3ed5aea2..b15874bd82 100644 --- a/tests/gdt/grids.list +++ b/tests/gdt/grids.list @@ -4,3 +4,4 @@ 004 export 005 presentation 006 view +007 notes diff --git a/tests/gdt/notes/A1 b/tests/gdt/notes/A1 new file mode 100644 index 0000000000..67316e0662 --- /dev/null +++ b/tests/gdt/notes/A1 @@ -0,0 +1,5 @@ + +# expected results +set nb_annotations_result 0 +set nb_notes_result 0 +set nb_orphan_result 0 diff --git a/tests/gdt/notes/A2 b/tests/gdt/notes/A2 new file mode 100644 index 0000000000..a9b0e61e4c --- /dev/null +++ b/tests/gdt/notes/A2 @@ -0,0 +1,13 @@ + +# create comment note +set comment [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment} + +# create balloon note +set balloon [XNoteCreateComment D "Hello, Everyone!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${balloon} + +# expected results +set nb_annotations_result 0 +set nb_notes_result 2 +set nb_orphan_result 2 diff --git a/tests/gdt/notes/A3 b/tests/gdt/notes/A3 new file mode 100644 index 0000000000..d51374cdb4 --- /dev/null +++ b/tests/gdt/notes/A3 @@ -0,0 +1,9 @@ + +# create binary note +set data [XNoteCreateBinData D "Binary data" --data "abcdefghijklmnopqrstuvwxyz" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${data} + +# expected results +set nb_annotations_result 0 +set nb_notes_result 1 +set nb_orphan_result 1 diff --git a/tests/gdt/notes/A4 b/tests/gdt/notes/A4 new file mode 100644 index 0000000000..ed25fabadc --- /dev/null +++ b/tests/gdt/notes/A4 @@ -0,0 +1,16 @@ + +# create comment note +set comment [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment} + +# create balloon note +set balloon [XNoteCreateComment D "Hello, Everyone!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${balloon} + +# delete comment note +XNoteDelete D ${comment} + +# expected results +set nb_annotations_result 0 +set nb_notes_result 1 +set nb_orphan_result 1 diff --git a/tests/gdt/notes/A5 b/tests/gdt/notes/A5 new file mode 100644 index 0000000000..08e910c21a --- /dev/null +++ b/tests/gdt/notes/A5 @@ -0,0 +1,16 @@ + +# create comment note +set comment [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment} + +# create balloon note +set balloon [XNoteCreateComment D "Hello, Everyone!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${balloon} + +# delete comment note +XNoteDeleteAll D + +# expected results +set nb_annotations_result 0 +set nb_notes_result 0 +set nb_orphan_result 0 diff --git a/tests/gdt/notes/B1 b/tests/gdt/notes/B1 new file mode 100644 index 0000000000..c79c0ab2cc --- /dev/null +++ b/tests/gdt/notes/B1 @@ -0,0 +1,18 @@ + +# create comment note +set comment [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment} + +# annotate box_1 +XNoteRefDump D [XNoteAdd D ${comment} ${box_1}] + +# annotate box_2 +XNoteRefDump D [XNoteAdd D ${comment} ${box_2}] + +# list annotations +XNoteAnnotations D + +# expected results +set nb_annotations_result 2 +set nb_notes_result 1 +set nb_orphan_result 0 diff --git a/tests/gdt/notes/B2 b/tests/gdt/notes/B2 new file mode 100644 index 0000000000..7172784bd9 --- /dev/null +++ b/tests/gdt/notes/B2 @@ -0,0 +1,22 @@ + +# create comment_1 note +set comment_1 [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_1} + +# create comment_2 note +set comment_2 [XNoteCreateComment D "Hello, Everyone!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_2} + +# annotate box_1 +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1}] + +# annotate box_2 +XNoteRefDump D [XNoteAdd D ${comment_2} ${box_2}] + +# list annotations +XNoteAnnotations D + +# expected results +set nb_annotations_result 2 +set nb_notes_result 2 +set nb_orphan_result 0 diff --git a/tests/gdt/notes/B3 b/tests/gdt/notes/B3 new file mode 100644 index 0000000000..d3990ebc12 --- /dev/null +++ b/tests/gdt/notes/B3 @@ -0,0 +1,25 @@ + +# create comment_1 note +set comment_1 [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_1} + +# create comment_2 note +set comment_2 [XNoteCreateComment D "Hello, Everyone!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_2} + +# annotate box_1 +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1}] + +# annotate box_2 +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_2}] + +# list annotations +XNoteAnnotations D + +# delete orphan notes +XNoteDeleteOrphan D + +# expected results +set nb_annotations_result 2 +set nb_notes_result 1 +set nb_orphan_result 0 diff --git a/tests/gdt/notes/B4 b/tests/gdt/notes/B4 new file mode 100644 index 0000000000..2adfe3f1cc --- /dev/null +++ b/tests/gdt/notes/B4 @@ -0,0 +1,22 @@ + +# create comment_1 note +set comment_1 [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_1} + +# create comment_2 note +set comment_2 [XNoteCreateComment D "Hello, Everyone!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_2} + +# annotate box_1 +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1}] + +# annotate box_1 +XNoteRefDump D [XNoteAdd D ${comment_2} ${box_1}] + +# list annotations +XNoteAnnotations D + +# expected results +set nb_annotations_result 1 +set nb_notes_result 2 +set nb_orphan_result 0 diff --git a/tests/gdt/notes/B5 b/tests/gdt/notes/B5 new file mode 100644 index 0000000000..43f9866d93 --- /dev/null +++ b/tests/gdt/notes/B5 @@ -0,0 +1,15 @@ + +# create comment note +set comment [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment} + +# annotate box_1 attribute +XNoteRefDump D [XNoteAdd D ${comment} ${box_1} --attr ${guid}] + +# list annotations +XNoteAnnotations D + +# expected results +set nb_annotations_result 1 +set nb_notes_result 1 +set nb_orphan_result 0 diff --git a/tests/gdt/notes/B6 b/tests/gdt/notes/B6 new file mode 100644 index 0000000000..92f81cd941 --- /dev/null +++ b/tests/gdt/notes/B6 @@ -0,0 +1,18 @@ + +# create comment note +set comment [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment} + +# annotate box_1 +XNoteRefDump D [XNoteAdd D ${comment} ${box_1}] + +# annotate box_1 attribute +XNoteRefDump D [XNoteAdd D ${comment} ${box_1} --attr ${guid}] + +# list annotations +XNoteAnnotations D + +# expected results +set nb_annotations_result 2 +set nb_notes_result 1 +set nb_orphan_result 0 diff --git a/tests/gdt/notes/B7 b/tests/gdt/notes/B7 new file mode 100644 index 0000000000..a6b102691f --- /dev/null +++ b/tests/gdt/notes/B7 @@ -0,0 +1,21 @@ + +# create comment note +set comment [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment} + +# annotate box_1 +XNoteRefDump D [XNoteAdd D ${comment} ${box_1}] + +# annotate box_1 attribute +XNoteRefDump D [XNoteAdd D ${comment} ${box_1} --attr ${guid}] + +# annotate box_1 subshape 1 +XNoteRefDump D [XNoteAdd D ${comment} ${box_1} --subshape 1] + +# list annotations +XNoteAnnotations D + +# expected results +set nb_annotations_result 3 +set nb_notes_result 1 +set nb_orphan_result 0 diff --git a/tests/gdt/notes/C1 b/tests/gdt/notes/C1 new file mode 100644 index 0000000000..cb85e0552a --- /dev/null +++ b/tests/gdt/notes/C1 @@ -0,0 +1,24 @@ + +# create comment note +set comment [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment} + +# annotate box_1 +XNoteRefDump D [XNoteAdd D ${comment} ${box_1}] + +# annotate box_2 +XNoteRefDump D [XNoteAdd D ${comment} ${box_2}] + +# list annotations +XNoteAnnotations D + +# delete note +XNoteDelete D ${comment} + +# list annotations +XNoteAnnotations D + +# expected results +set nb_annotations_result 0 +set nb_notes_result 0 +set nb_orphan_result 0 diff --git a/tests/gdt/notes/C2 b/tests/gdt/notes/C2 new file mode 100644 index 0000000000..e8be62e537 --- /dev/null +++ b/tests/gdt/notes/C2 @@ -0,0 +1,28 @@ + +# create comment_1 note +set comment_1 [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_1} + +# create comment_2 note +set comment_2 [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_2} + +# annotate box_1 +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1}] + +# annotate box_2 +XNoteRefDump D [XNoteAdd D ${comment_2} ${box_2}] + +# list annotations +XNoteAnnotations D + +# delete comment_1 note +XNoteDelete D ${comment_1} + +# list annotations +XNoteAnnotations D + +# expected results +set nb_annotations_result 1 +set nb_notes_result 1 +set nb_orphan_result 0 diff --git a/tests/gdt/notes/C3 b/tests/gdt/notes/C3 new file mode 100644 index 0000000000..37d47964ae --- /dev/null +++ b/tests/gdt/notes/C3 @@ -0,0 +1,28 @@ + +# create comment_1 note +set comment_1 [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_1} + +# create comment_2 note +set comment_2 [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_2} + +# annotate box_1 +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1}] + +# annotate box_2 +XNoteRefDump D [XNoteAdd D ${comment_2} ${box_2}] + +# list annotations +XNoteAnnotations D + +# remove comment_1 annotation +XNoteRemove D ${box_1} ${comment_1} + +# list annotations +XNoteAnnotations D + +# expected results +set nb_annotations_result 1 +set nb_notes_result 2 +set nb_orphan_result 1 diff --git a/tests/gdt/notes/C4 b/tests/gdt/notes/C4 new file mode 100644 index 0000000000..6b8b466685 --- /dev/null +++ b/tests/gdt/notes/C4 @@ -0,0 +1,27 @@ + +# create comment note +set comment [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment} + +# annotate box_1 +XNoteRefDump D [XNoteAdd D ${comment} ${box_1}] + +# annotate box_1 attribute +XNoteRefDump D [XNoteAdd D ${comment} ${box_1} --attr ${guid}] + +# annotate box_1 subshape 1 +XNoteRefDump D [XNoteAdd D ${comment} ${box_1} --subshape 1] + +# list annotations +XNoteAnnotations D + +# remove attribute annotation +XNoteRemove D ${box_1} ${comment} --attr ${guid} + +# remove subshape annotation +XNoteRemove D ${box_1} ${comment} --subshape 1 + +# expected results +set nb_annotations_result 1 +set nb_notes_result 1 +set nb_orphan_result 0 diff --git a/tests/gdt/notes/C5 b/tests/gdt/notes/C5 new file mode 100644 index 0000000000..8031a12c04 --- /dev/null +++ b/tests/gdt/notes/C5 @@ -0,0 +1,29 @@ + +# create comment_1 note +set comment_1 [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_1} + +# create comment_2 note +set comment_2 [XNoteCreateComment D "Hello, Everyone!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_2} + +# annotate box_1 +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1}] + +# annotate box_1 attribute +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1} --attr ${guid}] +XNoteRefDump D [XNoteAdd D ${comment_2} ${box_1} --attr ${guid}] + +# annotate box_1 subshape 1 +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1} --subshape 1] + +# list annotations +XNoteAnnotations D + +# remove all attribute annotations +XNoteRemoveAll D ${box_1} --attr ${guid} + +# expected results +set nb_annotations_result 2 +set nb_notes_result 2 +set nb_orphan_result 1 diff --git a/tests/gdt/notes/C6 b/tests/gdt/notes/C6 new file mode 100644 index 0000000000..4c3e71ace9 --- /dev/null +++ b/tests/gdt/notes/C6 @@ -0,0 +1,29 @@ + +# create comment_1 note +set comment_1 [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_1} + +# create comment_2 note +set comment_2 [XNoteCreateComment D "Hello, Everyone!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_2} + +# annotate box_1 +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1}] + +# annotate box_1 attribute +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1} --attr ${guid}] + +# annotate box_1 subshape 1 +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1} --subshape 1] +XNoteRefDump D [XNoteAdd D ${comment_2} ${box_1} --subshape 1] + +# list annotations +XNoteAnnotations D + +# remove all attribute annotations +XNoteRemoveAll D ${box_1} --subshape 1 + +# expected results +set nb_annotations_result 2 +set nb_notes_result 2 +set nb_orphan_result 1 diff --git a/tests/gdt/notes/C7 b/tests/gdt/notes/C7 new file mode 100644 index 0000000000..0aa7749492 --- /dev/null +++ b/tests/gdt/notes/C7 @@ -0,0 +1,31 @@ + +# create comment_1 note +set comment_1 [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_1} + +# create comment_2 note +set comment_2 [XNoteCreateComment D "Hello, Everyone!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_2} + +# annotate box_1 +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1}] +XNoteRefDump D [XNoteAdd D ${comment_2} ${box_1}] + +# annotate box_1 attribute +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1} --attr ${guid}] +XNoteRefDump D [XNoteAdd D ${comment_2} ${box_1} --attr ${guid}] + +# annotate box_1 subshape 1 +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1} --subshape 1] +XNoteRefDump D [XNoteAdd D ${comment_2} ${box_1} --subshape 1] + +# list annotations +XNoteAnnotations D + +# remove all box annotations +XNoteRemoveAll D ${box_1} + +# expected results +set nb_annotations_result 2 +set nb_notes_result 2 +set nb_orphan_result 0 diff --git a/tests/gdt/notes/C8 b/tests/gdt/notes/C8 new file mode 100644 index 0000000000..85f41d9014 --- /dev/null +++ b/tests/gdt/notes/C8 @@ -0,0 +1,33 @@ + +# create comment_1 note +set comment_1 [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_1} + +# create comment_2 note +set comment_2 [XNoteCreateComment D "Hello, Everyone!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_2} + +# annotate box_1 +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1}] +XNoteRefDump D [XNoteAdd D ${comment_2} ${box_1}] + +# annotate box_1 attribute +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1} --attr ${guid}] +XNoteRefDump D [XNoteAdd D ${comment_2} ${box_1} --attr ${guid}] + +# annotate box_1 subshape 1 +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1} --subshape 1] +XNoteRefDump D [XNoteAdd D ${comment_2} ${box_1} --subshape 1] + +# list annotations +XNoteAnnotations D + +# remove all box annotations +XNoteRemoveAll D ${box_1} +XNoteRemoveAll D ${box_1} --attr ${guid} +XNoteRemoveAll D ${box_1} --subshape 1 + +# expected results +set nb_annotations_result 0 +set nb_notes_result 2 +set nb_orphan_result 2 diff --git a/tests/gdt/notes/C9 b/tests/gdt/notes/C9 new file mode 100644 index 0000000000..8b7a49fb19 --- /dev/null +++ b/tests/gdt/notes/C9 @@ -0,0 +1,33 @@ + +# create comment_1 note +set comment_1 [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_1} + +# create comment_2 note +set comment_2 [XNoteCreateComment D "Hello, Everyone!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_2} + +# annotate box_1 +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1}] +XNoteRefDump D [XNoteAdd D ${comment_2} ${box_1}] + +# annotate box_1 attribute +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1} --attr ${guid}] +XNoteRefDump D [XNoteAdd D ${comment_2} ${box_1} --attr ${guid}] + +# annotate box_1 subshape 1 +XNoteRefDump D [XNoteAdd D ${comment_1} ${box_1} --subshape 1] +XNoteRefDump D [XNoteAdd D ${comment_2} ${box_1} --subshape 1] + +# list annotations +XNoteAnnotations D + +# remove all box annotations +XNoteRemoveAll D ${box_1} --del-orphan +XNoteRemoveAll D ${box_1} --attr ${guid} --del-orphan +XNoteRemoveAll D ${box_1} --subshape 1 --del-orphan + +# expected results +set nb_annotations_result 0 +set nb_notes_result 0 +set nb_orphan_result 0 diff --git a/tests/gdt/notes/D1 b/tests/gdt/notes/D1 new file mode 100644 index 0000000000..49bf36c6df --- /dev/null +++ b/tests/gdt/notes/D1 @@ -0,0 +1,51 @@ + +# create comment_1 note +set comment_1 [XNoteCreateComment D "Hello, World!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_1} + +# create comment_2 note +set comment_2 [XNoteCreateComment D "Hello, Everyone!" --user "The user" --time [clock format [clock seconds] -format "%Y-%m-%dT%H:%M:%S"]] +XNoteDump D ${comment_2} + +# annotate box_1 +set annotated_1 [XNoteAdd D ${comment_1} ${box_1}] +set annotated_2 [XNoteAdd D ${comment_2} ${box_1}] +if {${annotated_1} != ${annotated_2}} { + puts "Error : item annotation labels (${annotated_1} and ${annotated_2}) must be equal not equal." +} + +# annotate box_1 attribute +set annotated_1_attr [XNoteAdd D ${comment_1} ${box_1} --attr ${guid}] +set annotated_2_attr [XNoteAdd D ${comment_2} ${box_1} --attr ${guid}] +if {${annotated_1_attr} != ${annotated_2_attr}} { + puts "Error : item's attribute annotation labels (${annotated_1_attr} and ${annotated_2_attr}) must be equal not equal." +} + +# annotate box_1 subshape 1 +set annotated_1_subshape [XNoteAdd D ${comment_1} ${box_1} --subshape 1] +set annotated_2_subshape [XNoteAdd D ${comment_2} ${box_1} --subshape 1] +if {${annotated_1_subshape} != ${annotated_2_subshape}} { + puts "Error : item's subshape annotation labels (${annotated_1_subshape} and ${annotated_2_subshape}) must be equal not equal." +} + +# find annotations +set annotated_found [XNoteFindAnnotated D ${box_1}] +if {${annotated_1} != ${annotated_found}} { + puts "Error : ${annotated_1} is expected to found instead of ${annotated_found}" +} +set annotated_attr_found [XNoteFindAnnotated D ${box_1} --attr ${guid}] +if {${annotated_1_attr} != ${annotated_attr_found}} { + puts "Error : ${annotated_1_attr} is expected to found instead of ${annotated_attr_found}" +} +set annotated_subshape_found [XNoteFindAnnotated D ${box_1} --subshape 1] +if {${annotated_1_subshape} != ${annotated_subshape_found}} { + puts "Error : ${annotated_1_subshape} is expected to found instead of ${annotated_subshape_found}" +} + +# list annotations +XNoteAnnotations D + +# expected results +set nb_annotations_result 3 +set nb_notes_result 2 +set nb_orphan_result 0 diff --git a/tests/gdt/notes/begin b/tests/gdt/notes/begin new file mode 100644 index 0000000000..96fdf5d182 --- /dev/null +++ b/tests/gdt/notes/begin @@ -0,0 +1,21 @@ + +# assembly of 2 boxes +box b1 1 1 1 +box b2 2 2 2 +btranslate b1 b1 2 2 2 +btranslate b2 b2 -3 -3 -3 +compound b1 b2 c + +# new document +NewDocument D + +# populate D +XAddShape D c + +# box extries +set box_1 "0:1:1:2" +set box_2 "0:1:1:3" + +# add int attribute to box_1 +set guid "27927843-72A0-41ef-901D-621FAB01C27A" +SetInteger D ${box_1} 1965 ${guid} diff --git a/tests/gdt/notes/end b/tests/gdt/notes/end new file mode 100644 index 0000000000..48d26b9c2f --- /dev/null +++ b/tests/gdt/notes/end @@ -0,0 +1,20 @@ + +# check counts +set count [XNoteCount D] +set nb_annotations [lindex ${count} 0] +set nb_notes [lindex ${count} 1] +set nb_orphan [lindex ${count} 2] +if {${nb_annotations} != ${nb_annotations_result}} { + puts "Error : ${nb_annotations_result} number of annotated items is expected instead of ${nb_annotations}!" +} +if {${nb_notes} != ${nb_notes_result}} { + puts "Error : ${nb_notes_result} number of notes is expected instead of ${nb_notes}!" +} +if {${nb_orphan} != ${nb_orphan_result}} { + puts "Error : ${nb_orphan_result} number of orphan notes is expected instead of ${nb_orphan}!" +} + +# release document +unset D + +puts "TEST COMPLETED"