From d7d20bee619bd7cc8dcda329957a437d66440d7b Mon Sep 17 00:00:00 2001 From: Sebastian Castro Date: Sat, 25 Nov 2023 12:26:09 -0500 Subject: [PATCH 1/6] Add frame visualization to Meshcat visualizer --- .../pinocchio/visualize/meshcat_visualizer.py | 44 +++++++++++++++++++ 1 file changed, 44 insertions(+) diff --git a/bindings/python/pinocchio/visualize/meshcat_visualizer.py b/bindings/python/pinocchio/visualize/meshcat_visualizer.py index c8f7c75454..2052ee4876 100644 --- a/bindings/python/pinocchio/visualize/meshcat_visualizer.py +++ b/bindings/python/pinocchio/visualize/meshcat_visualizer.py @@ -20,6 +20,14 @@ } COLOR_PRESETS = DEFAULT_COLOR_PROFILES.copy() +FRAME_AXIS_POSITIONS = np.array([ + [0, 0, 0], [1, 0, 0], + [0, 0, 0], [0, 1, 0], + [0, 0, 0], [0, 0, 1]]).astype(np.float32).T +FRAME_AXIS_COLORS = np.array([ + [1, 0, 0], [1, 0.6, 0], + [0, 1, 0], [0.6, 1, 0], + [0, 0, 1], [0, 0.6, 1]]).astype(np.float32).T def isMesh(geometry_object): """Check whether the geometry object contains a Mesh supported by MeshCat""" @@ -485,6 +493,9 @@ def loadViewerModel(self, rootNodeName="pinocchio", color=None): # Visuals self.viewerVisualGroupName = self.viewerRootNodeName + "/" + "visuals" + # Frames + self.viewerFramesGroupName = self.viewerRootNodeName + "/" + "frames" + for visual in self.visual_model.geometryObjects: self.loadViewerGeometryObject(visual, pin.GeometryType.VISUAL, color) self.displayVisuals(True) @@ -596,6 +607,39 @@ def displayVisuals(self, visibility): if visibility: self.updatePlacements(pin.GeometryType.VISUAL) + def drawFrame( + self, frame_id, axis_length=0.2, axis_width=2 + ): + import meshcat.geometry as mg + frame_name = self.model.frames[frame_id].name + frame_viz_name = f"{self.viewerFramesGroupName}/{frame_name}" + self.viewer[frame_viz_name].set_object( + mg.LineSegments( + mg.PointsGeometry( + position=axis_length * FRAME_AXIS_POSITIONS, + color=FRAME_AXIS_COLORS, + ), + mg.LineBasicMaterial( + linewidth=axis_width, + vertexColors=True, + ), + ) + ) + self.viewer[frame_viz_name].set_transform( + self.data.oMf[frame_id].homogeneous + ) + + def drawFrames( + self, frame_ids=None, axis_length=0.2, axis_width=2 + ): + for fid, _ in enumerate(self.model.frames): + if frame_ids is None or fid in frame_ids: + self.drawFrame( + fid, + axis_length=axis_length, + axis_width=axis_width, + ) + def drawFrameVelocities( self, frame_id, v_scale=0.2, color=FRAME_VEL_COLOR ): # pylint: disable=arguments-differ From eb853d091e59b0915c3c191aada8e38573cd02a7 Mon Sep 17 00:00:00 2001 From: Sebastian Castro Date: Sat, 25 Nov 2023 12:44:46 -0500 Subject: [PATCH 2/6] Remove trailing whitespace --- bindings/python/pinocchio/visualize/meshcat_visualizer.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/bindings/python/pinocchio/visualize/meshcat_visualizer.py b/bindings/python/pinocchio/visualize/meshcat_visualizer.py index 7eccde22ca..979e6cc546 100644 --- a/bindings/python/pinocchio/visualize/meshcat_visualizer.py +++ b/bindings/python/pinocchio/visualize/meshcat_visualizer.py @@ -621,7 +621,7 @@ def drawFrame( self.viewer[frame_viz_name].set_transform( self.data.oMf[frame_id].homogeneous ) - + def drawFrames( self, frame_ids=None, axis_length=0.2, axis_width=2 ): From 1186f623b8b024a514e0b6fd7a97cf06b0690851 Mon Sep 17 00:00:00 2001 From: Sebastian Castro Date: Sat, 25 Nov 2023 14:16:18 -0500 Subject: [PATCH 3/6] Separate initialization and update of frames --- .../pinocchio/visualize/meshcat_visualizer.py | 75 +++++++++++-------- 1 file changed, 43 insertions(+), 32 deletions(-) diff --git a/bindings/python/pinocchio/visualize/meshcat_visualizer.py b/bindings/python/pinocchio/visualize/meshcat_visualizer.py index 979e6cc546..051227b8f5 100644 --- a/bindings/python/pinocchio/visualize/meshcat_visualizer.py +++ b/bindings/python/pinocchio/visualize/meshcat_visualizer.py @@ -485,14 +485,14 @@ def loadViewerModel(self, rootNodeName="pinocchio", color=None): # Visuals self.viewerVisualGroupName = self.viewerRootNodeName + "/" + "visuals" - - # Frames - self.viewerFramesGroupName = self.viewerRootNodeName + "/" + "frames" - for visual in self.visual_model.geometryObjects: self.loadViewerGeometryObject(visual, pin.GeometryType.VISUAL, color) self.displayVisuals(True) + # Frames + self.viewerFramesGroupName = self.viewerRootNodeName + "/" + "frames" + self.displayFrames(False) + def reload(self, new_geometry_object, geometry_type=None): """Reload a geometry_object given by its name and its type""" if geometry_type == pin.GeometryType.VISUAL: @@ -526,6 +526,9 @@ def display(self, q=None): if self.display_visuals: self.updatePlacements(pin.GeometryType.VISUAL) + if self.display_frames: + self.updateFrames() + def updatePlacements(self, geometry_type): if geometry_type == pin.GeometryType.VISUAL: geom_model = self.visual_model @@ -600,38 +603,46 @@ def displayVisuals(self, visibility): if visibility: self.updatePlacements(pin.GeometryType.VISUAL) - def drawFrame( - self, frame_id, axis_length=0.2, axis_width=2 - ): + def displayFrames(self, visibility, frame_ids=None, axis_length=0.2, axis_width=2): + """Set whether to display frames or not.""" + self.display_frames = visibility + if visibility: + self.initializeFrames(frame_ids, axis_length, axis_width) + self.viewer[self.viewerFramesGroupName].set_property("visible", visibility) + + def initializeFrames(self, frame_ids=None, axis_length=0.2, axis_width=2): + """Initializes the frame objects for display.""" import meshcat.geometry as mg - frame_name = self.model.frames[frame_id].name - frame_viz_name = f"{self.viewerFramesGroupName}/{frame_name}" - self.viewer[frame_viz_name].set_object( - mg.LineSegments( - mg.PointsGeometry( - position=axis_length * FRAME_AXIS_POSITIONS, - color=FRAME_AXIS_COLORS, - ), - mg.LineBasicMaterial( - linewidth=axis_width, - vertexColors=True, - ), - ) - ) - self.viewer[frame_viz_name].set_transform( - self.data.oMf[frame_id].homogeneous - ) + self.viewer[self.viewerFramesGroupName].delete() + self.frame_ids = [] - def drawFrames( - self, frame_ids=None, axis_length=0.2, axis_width=2 - ): - for fid, _ in enumerate(self.model.frames): + for fid, frame in enumerate(self.model.frames): if frame_ids is None or fid in frame_ids: - self.drawFrame( - fid, - axis_length=axis_length, - axis_width=axis_width, + frame_viz_name = f"{self.viewerFramesGroupName}/{frame.name}" + self.viewer[frame_viz_name].set_object( + mg.LineSegments( + mg.PointsGeometry( + position=axis_length * FRAME_AXIS_POSITIONS, + color=FRAME_AXIS_COLORS, + ), + mg.LineBasicMaterial( + linewidth=axis_width, + vertexColors=True, + ), + ) ) + self.frame_ids.append(fid) + + def updateFrames(self): + """ + Updates the frame visualizations with the latest transforms from model data. + """ + for fid in self.frame_ids: + frame_name = self.model.frames[fid].name + frame_viz_name = f"{self.viewerFramesGroupName}/{frame_name}" + self.viewer[frame_viz_name].set_transform( + self.data.oMf[fid].homogeneous + ) def drawFrameVelocities( self, frame_id, v_scale=0.2, color=FRAME_VEL_COLOR From db5233caab3e675d2496a62ac3b60c5e22ec6558 Mon Sep 17 00:00:00 2001 From: Sebastian Castro Date: Sat, 25 Nov 2023 14:32:28 -0500 Subject: [PATCH 4/6] Update frames in FK if needed --- bindings/python/pinocchio/visualize/meshcat_visualizer.py | 1 + 1 file changed, 1 insertion(+) diff --git a/bindings/python/pinocchio/visualize/meshcat_visualizer.py b/bindings/python/pinocchio/visualize/meshcat_visualizer.py index 051227b8f5..23c9e48fa6 100644 --- a/bindings/python/pinocchio/visualize/meshcat_visualizer.py +++ b/bindings/python/pinocchio/visualize/meshcat_visualizer.py @@ -637,6 +637,7 @@ def updateFrames(self): """ Updates the frame visualizations with the latest transforms from model data. """ + pin.updateFramePlacements(self.model, self.data) for fid in self.frame_ids: frame_name = self.model.frames[fid].name frame_viz_name = f"{self.viewerFramesGroupName}/{frame_name}" From a109f145125466c965c8e5887169be0518d7c1d3 Mon Sep 17 00:00:00 2001 From: Justin Carpentier Date: Sun, 26 Nov 2023 12:03:08 +0100 Subject: [PATCH 5/6] changelog: update --- CHANGELOG.md | 1 + 1 file changed, 1 insertion(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 4ca3cea695..6ce59cbba0 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -9,6 +9,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/). ### Added - Add inverse dynamics (`rnea`) Python and C++ example ([#2083](https://github.com/stack-of-tasks/pinocchio/pull/2083)) +- Add visualization of Frames in MeshCat viewer ([#2098](https://github.com/stack-of-tasks/pinocchio/pull/2098)) ### Fixed From 4ccbcf2fb6dee875543b6fb90fb563763bda1716 Mon Sep 17 00:00:00 2001 From: Justin Carpentier Date: Sun, 26 Nov 2023 12:06:31 +0100 Subject: [PATCH 6/6] readme: add Sebastian Castro as contributor --- README.md | 1 + 1 file changed, 1 insertion(+) diff --git a/README.md b/README.md index 6baca0cd67..58cc2f391d 100644 --- a/README.md +++ b/README.md @@ -220,6 +220,7 @@ The following people have been involved in the development of **Pinocchio** and - [Shubham Singh](https://github.com/shubhamsingh91) (UT Austin): second-order inverse dynamics derivatives - [Stéphane Caron](https://scaron.info) (Inria): core developper - [Joris Vaillant](https://github.com/jorisv) (Inria): core developer and manager of the project +- [Sebastian Castro](https://roboticseabass.com) (PickNik Robotics): MeshCat viewer features extension If you have participated in the development of **Pinocchio**, please add your name and contribution to this list.