-
Notifications
You must be signed in to change notification settings - Fork 34
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add a parameter set selector and append the result in a table. #133
base: master
Are you sure you want to change the base?
Conversation
Hello @lassoan , Please consider this PR with the main purpose of adding a parameter node to the StenosisMeasurement3D module. The parameter node is mainly based on vtkMRMLCropVolumeParametersNode. Waiting for your comments. Regards. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks good overall, I don't have any problems if you merge it as is. But I added a couple of comments inline that would improve the design a bit.
StenosisMeasurement3D/qSlicerStenosisMeasurement3DModuleWidget.cxx
Outdated
Show resolved
Hide resolved
StenosisMeasurement3D/MRML/vtkMRMLStenosisMeasurement3DParameterNode.cxx
Outdated
Show resolved
Hide resolved
StenosisMeasurement3D/MRML/vtkMRMLStenosisMeasurement3DParameterNode.cxx
Outdated
Show resolved
Hide resolved
void operator=(const vtkMRMLStenosisMeasurement3DParameterNode&); | ||
|
||
std::string InputSegmentID; | ||
vtkWeakPointer<vtkPolyData> OutputWallOpenPolyData; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think vtkSetObjectMacro/vtkGetObjectMacro is compatible with vtkWeakPointer, but even if it does not crash or behaves incorrectly, it just does not make sense.
If you use vtkSetObjectMacro/vtkGetObjectMacro then it means that vtkMRMLStenosisMeasurement3DParameterNode owns the object (the parameter node keeps a reference to the object, so the object will not be deleted until the parameter node releases it) and the vtkMRMLStenosisMeasurement3DParameterNode should use a simple pointer (that is initialized to nullptr
here).
If you use a vtkSmartPointer then you can use vtkSetMacro/vtkGetMacro and the parameter node owns the object.
If you use a vtkWeakPointer then you can use vtkSetMacro/vtkGetMacro and some other object must keep a reference to the object.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Changed for vtkSmartPointer.
But vtkGet/SetMacro does not compile.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What is the error message? If adding #include <vtkSmartPointer.h>
does not solve the issue then you can investigate further (copy the error message here, maybe I have an idea). But often we don't need get/set functions, as you can set the variable with a single operator=.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Here is a typical error message:
In file included from /home/user/tmp/Slicer/SlicerExtension-VMTK/StenosisMeasurement3D/MRML/vtkMRMLStenosisMeasurement3DParameterNode.cxx:1:
/home/user/tmp/Slicer/SlicerExtension-VMTK/StenosisMeasurement3D/MRML/vtkMRMLStenosisMeasurement3DParameterNode.h:62:5: error: no viable overloaded '='
62 | vtkSetMacro(OutputWallOpenPolyData, vtkPolyData);
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
/home/arc/src/Slicer-SuperBuild9/VTK/Common/Core/vtkSetGet.h:134:18: note: expanded from macro 'vtkSetMacro'
134 | this->name = _arg;
| ~~~~~~~~~~ ^ ~~~~
/home/arc/src/Slicer-SuperBuild9/VTK/Common/Core/vtkSmartPointer.h:144:20: note: candidate function not viable: no known conversion from 'vtkPolyData' to 'const vtkSmartPointer' for 1st argument
144 | vtkSmartPointer& operator=(const vtkSmartPointer& r)
| ^ ~~~~~~~~~~~~~~~~~~~~~~~~
/home/arc/src/Slicer-SuperBuild9/VTK/Common/Core/vtkSmartPointer.h:151:20: note: candidate template ignored: could not match 'vtkSmartPointer' against 'vtkPolyData'
151 | vtkSmartPointer& operator=(const vtkSmartPointer& r)
| ^
/home/arc/src/Slicer-SuperBuild9/VTK/Common/Core/vtkSmartPointer.h:165:20: note: candidate template ignored: could not match 'vtkNew' against 'vtkPolyData'
165 | vtkSmartPointer& operator=(const vtkNew& r)
| ^
/home/arc/src/Slicer-SuperBuild9/VTK/Common/Core/vtkSmartPointer.h:178:20: note: candidate template ignored: could not match 'U ' against 'vtkPolyData'
178 | vtkSmartPointer& operator=(U r)
| ^
In file included from /home/user/tmp/Slicer/SlicerExtension-VMTK/StenosisMeasurement3D/MRML/vtkMRMLStenosisMeasurement3DParameterNode.cxx:1:
/home/user/tmp/Slicer/SlicerExtension-VMTK/StenosisMeasurement3D/MRML/vtkMRMLStenosisMeasurement3DParameterNode.h:63:5: error: no viable conversion from returned value of type 'vtkSmartPointer' to function return type 'vtkPolyData'
63 | vtkGetMacro(OutputWallOpenPolyData, vtkPolyData)
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
/home/arc/src/Slicer-SuperBuild9/VTK/Common/Core/vtkSetGet.h:156:12: note: expanded from macro 'vtkGetMacro'
156 | return this->name;
| ^~~~~~~~~~
/home/arc/src/Slicer-SuperBuild9/VTK/Common/DataModel/vtkPolyData.h:757:3: note: candidate constructor not viable: no known conversion from 'vtkSmartPointer' to 'const vtkPolyData &' for 1st argument
757 | vtkPolyData(const vtkPolyData&) = delete;
| ^ ~~~~~~~~~~~~~~~~~~
/home/arc/src/Slicer-SuperBuild9/VTK/Common/Core/vtkSmartPointer.h:197:3: note: candidate function
197 | operator T*() const noexcept { return static_cast<T*>(this->Object); }
| ^
5 errors generated.
I'll probably not use the macro.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You may need to add #include <vtkPolyData.h>
as well.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
vtkPolyData.h
is already included. vtkSmartPointer.h
and vtkSetGet.h
do not help.
I'm using clang version 18.1.8
, don't know if it's the problem.
Dropping the macro is probably the simplest solution.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
vtkSet/GetObjectMacro is used when the VTK object is stored as a simple pointer. I'm not sure it is compatible with vtkSmartPointer.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should we then use the like of
vtkPolyData* GetOutputWallOpenPolyData() {return this->OutputWallOpenPolyData;}
instead of
vtkGetObjectMacro(OutputWallOpenPolyData, vtkPolyData);
?
It's an easy rewrite, I just need instructions.
|
||
QObject::connect(d->parameterSetSelector, SIGNAL(currentNodeChanged(vtkMRMLNode*)), | ||
this, SLOT(setParameterNode(vtkMRMLNode*))); | ||
|
||
// Put p1 and p2 ficucial points on the tube spline at nearest point when they are moved. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We always need to consider the GUI as optional. The module should be fully functional without instantiating a GUI widget. Therefore, you need to put in the logic things like observing some node and updating the scene content when the observed node is modified.
There are convenience features in module logic classes that make it relatively simple to observe modifications of all nodes of a certain type (see vtkSetAndObserveMRMLNodeEventsMacro
and ProcessMRMLNodesEvents
and how it is used for example in vtkSlicerMarkupsLogic
).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Using vtkObserveMRMLNodeEventsMacro
in ProcessMRMLNodesEvents
How does it differ from vtkSetAndObserveMRMLNodeEventsMacro
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
vtkSetAndObserveMRMLNodeEventsMacro is useful when you want to keep a pointer to a node and observe events that the node invokes. The macro does everything you need: removes observations from the previously observed node, releases reference from the old node, adds observations and reference to the newly selected node.
vtkObserveMRMLNodeEventsMacro just adds observations to a node.
StenosisMeasurement3D/qSlicerStenosisMeasurement3DModuleWidget.cxx
Outdated
Show resolved
Hide resolved
StenosisMeasurement3D/qSlicerStenosisMeasurement3DModuleWidget.cxx
Outdated
Show resolved
Hide resolved
StenosisMeasurement3D/qSlicerStenosisMeasurement3DModuleWidget.cxx
Outdated
Show resolved
Hide resolved
Thank you for this insightful review, I'll fix all these. |
a3d0ba9
to
44945da
Compare
Hi @lassoan , I made all the changes, moving much code in logic. Testing from Python is successful. I'm ready to change other things per your instructions. Thanks and regards. |
Great! I'll review your changes in details, so if you have any specific questions then let me know. |
ff1e4a6
to
3ec9bfa
Compare
ebfab15
to
028bb5a
Compare
86f9093
to
059366d
Compare
059366d
to
75c62be
Compare
StenosisMeasurement3D - create an MRML parameter node class - do not create a default parameter node in the selector - enforce selection of a parameter set - use the parameter node in logic - append the results in an MRML table - improve existing code.
75c62be
to
6ddb8f0
Compare
StenosisMeasurement3D