Search code examples
pythonnumpymathvtkquaternions

Python - VTK limit rotation of object issue (Quaternion)


I am trying to limit the rotation of the boxwidget by interaction, to enable the user to just rotate the box around the up-axis.

Therefore I have tried the approach to add an observer to the "InteractionEvent", and use the call-back function to reverse the rotation around the x- and z-axis.

After implementation I have faced two problems:

-) The changes on the transformation of the object - had no visible effect in the window (Maybe I am just working with the copy of the 3D-object?)

-) The calculation which I made, changed the quaternion - but not as assumed. It was mentioned to just negate the sign of x and z and keep the degree the same. I hoped to see as result of the modified rotation that the x and z values become zero... I assume that I misunderstood quaternions in general, or that I made a mistake with the vtk function calls

Old Orientation                                   [11.79730158  0.87192947  0.29194216  0.39307604]
Will rotate object with the following quaternion: [11.79730158 -0.87192947 -0.         -0.39307604]
New Orientation                                   [ 3.47624906 -0.17562846  0.98392544  0.03233235]
Different Access: Orientation                     [ 3.47624906 -0.17562846  0.98392544  0.03233235]

The results are produced by the following code:

import numpy as np
import vtk
import math
from math import atan2, sqrt, pi
from vtk.util import numpy_support

##########################################################################
def transform(T, points):
    assert(points.shape[0] >= 3)
    assert(T.shape == (4, 4))
    points_hom = np.vstack((points[0:3], np.ones((1, points.shape[1]))))
    points_hom = np.matmul(T, points_hom)
    points_hom = points_hom[0:3, :] / points_hom[3, :]
    points = np.vstack((points_hom, points[3:, :]))
    return points


def LeftButtonPressEvent(obj, event):
    t = vtk.vtkTransform()
    obj.GetTransform(t)
    q = np.zeros(4)
    t.GetOrientationWXYZ(q)
    print(f"Old Orientation {q}")
    q[2] = 0
    q[1:] = -q[1:]
    #angle = 2.0 * atan2(np.linalg.norm(q[1:]), q[0]/180.0*pi)*180.0/pi
    print(f"Will rotate object with the following quaternion: {q}")
    t.RotateWXYZ(q[0], q[1:])
    obj.GetProp3D().SetUserTransform( t )
    t.GetOrientationWXYZ(q)
    print(f"New Orientation {q}")
    obj_transform = obj.GetProp3D().GetUserTransform()
    obj_transform.GetOrientationWXYZ(q)
    print(f"Different Access: Orientation {q}\n\n\n")


##########################################################################
def init_vtk_pointcloud(pts, point_size=2.0, points_per_row=True):
    if not points_per_row:
        pts = pts.T
    assert (pts.shape[1] == 3 or
            pts.shape[1] == 4 or
            pts.shape[1] == 6)
    if pts.shape[1] == 3:
        color = np.ones((pts.shape[0], 3), np.uint8) * 255
    elif pts.shape[1] == 4:
        color = (pts[:, 3] * np.ones((3, 1)) * 255).astype(np.uint8).T
    elif pts.shape[1] == 6 or pts.shape[1] == 7:
        color = pts[:, 3:6]
    else:
        assert False, "Dimension of points is not 3 (without color), " \
                      "4 (with intensity) or 6/7 (with color and optionally intensity)"
    vtk_point_cloud = VtkPointCloud(point_size=point_size)
    vtk_point_cloud.set_points(pts[:, :3], color)
    return vtk_point_cloud


##########################################################################
def init_rendering(axes_scaling, background_color, camera, point_size, other_actors,
                   points_per_row, pts, window_name, window_size):
    vtk_point_cloud = init_vtk_pointcloud(pts, point_size, points_per_row)

    # Renderer
    renderer = vtk.vtkRenderer()
    renderer.SetActiveCamera(camera)
    renderer.SetBackground(*background_color)

    # Render Window
    render_window = vtk.vtkRenderWindow()
    render_window.AddRenderer(renderer)
    render_window.SetWindowName(window_name)
    render_window.SetSize(*window_size)

    # Add actor
    renderer.AddActor(vtk_point_cloud.vtkActor)
    return render_window, vtk_point_cloud


##########################################################################
def show_pointcloud(pts, window_name='Debug', points_per_row=True, background_color=[.0, .3, .4], camera=None,
                    axes_scaling=1.0, point_size=1.0, other_actors=[], window_size=(1280, 720), finalize=True):
    renderWindow, vtk_point_cloud = init_rendering(axes_scaling, background_color, camera, point_size,
                                                   other_actors, points_per_row, pts, window_name, window_size)

    #  Interactor
    renderWindowInteractor = vtk.vtkRenderWindowInteractor()
    renderWindowInteractor.SetInteractorStyle(
        PointSizeInteractorStyle(renderWindowInteractor, vtk_point_cloud))
    renderWindowInteractor.SetRenderWindow(renderWindow)

    boxActor = vtk.vtkActor()
    boxWidget = vtk.vtkBoxWidget()
    boxWidget.SetInteractor(renderWindowInteractor)#renderer interactor
    boxWidget.SetProp3D(boxActor)#actor here
    boxWidget.SetPlaceFactor(1.0)
    boxWidget.PlaceWidget()

    #ToDo: first interaction does not work except user scrolls beforehand... why?

    boxWidget.On()
    boxWidget.AddObserver("InteractionEvent", LeftButtonPressEvent)
    renderWindow.Render()
    renderWindowInteractor.Start()

    if finalize:
        renderWindow.Finalize()


##########################################################################
def create_cylinder(radius=1.0, height=1.0, pos=[0.0, 0.0, 0.0], approx_samples=1000):
    perimeter = 2 * math.pi * radius
    base_area = radius**2 * math.pi
    area = perimeter * height + 2*base_area

    # points per areal unit
    pts_per_au = approx_samples / area

    no_base_square_samples = int(pts_per_au*4*radius**2)
    base = np.random.rand(no_base_square_samples, 2) * 2 * radius - radius
    radii = np.linalg.norm(base, axis=1)
    mask = radii < radius
    base = base[mask, :]

    max_dim = max(perimeter, height)
    surface_samples = int(pts_per_au * (max_dim**2))

    surface = np.random.rand(surface_samples, 2) * max_dim
    mask = (surface[:, 0] < perimeter) & (surface[:, 1] < height)
    surface = surface[mask, :]
    surface = np.vstack((radius*np.sin(surface[:, 0]/radius), radius*np.cos(surface[:, 0]/radius),
                         surface[:, 1] - height/2.0)).T
    top = np.hstack((base, np.ones((base.shape[0], 1)) * height/2.0))
    bottom = top * (1.0, 1.0, -1.0)
    pts = np.vstack((top, bottom, surface)) + pos
    return pts



##########################################################################
class VtkPointCloud:

    def add_point(self, point):
        if self.vtkPoints.GetNumberOfPoints() < self.maxNumPoints:
            pointId = self.vtkPoints.InsertNextPoint(point[:])
            self.vtkDepth.InsertNextValue(point[2])
            self.vtkCells.InsertNextCell(1)
            self.vtkCells.InsertCellPoint(pointId)
        else:
            r = np.random.randint(0, self.maxNumPoints)
            self.vtkPoints.SetPoint(r, point[:])
        self.vtkCells.Modified()
        self.vtkPoints.Modified()
        self.vtkDepth.Modified()

    ##########################################################################
    def __init__(self, zMin=-10.0, zMax=10.0, maxNumPoints=1e6, point_size=2):
        self.maxNumPoints = maxNumPoints
        self.vtkPolyData = vtk.vtkPolyData()
        self.clear_points()
        mapper = vtk.vtkPolyDataMapper()
        mapper.SetInputData(self.vtkPolyData)
        mapper.SetColorModeToDefault()
        mapper.SetScalarRange(zMin, zMax)
        mapper.SetScalarVisibility(1)
        self.vtkActor = vtk.vtkActor()
        self.vtkActor.SetMapper(mapper)
        self.vtkActor.GetProperty().SetPointSize(point_size)

    ##########################################################################
    def set_points(self, points, color=None):
        if color is None:
            if points.shape[1] == 3:
                color = np.ones((points.shape[0], 3), np.uint8) * 255
            elif points.shape[1] == 4:
                color = value2hsvcolormap(points[:, 3])
                # color = (points[:, 3] * np.ones((3, 1)) * 255).astype(np.uint8).T
            elif points.shape[1] == 6 or points.shape[1] == 7:
                color = points[:, 3:6]

        color = np.ascontiguousarray(color)
        points = np.ascontiguousarray(points[:, :3])

        vtk_cell_idc = np.zeros((points.shape[0] * 2,), dtype=np.int64)
        vtk_cell_idc[0::2] = 1
        vtk_cell_idc[1::2] = np.arange(points.shape[0])

        vtk_point_data = numpy_support.numpy_to_vtk(num_array=points, deep=True, array_type=vtk.VTK_FLOAT)
        self.vtkPoints.SetData(vtk_point_data)

        if color is not None:
            vtk_color_data = numpy_support.numpy_to_vtk(num_array=color, deep=True, array_type=vtk.VTK_UNSIGNED_CHAR)
            vtk_color_data.SetName('Color')
            self.vtkPolyData.GetPointData().SetScalars(vtk_color_data)
            self.vtkPolyData.GetPointData().SetActiveScalars('Color')

        vtk_cell_data = numpy_support.numpy_to_vtk(num_array=vtk_cell_idc, deep=True, array_type=vtk.VTK_ID_TYPE)
        self.vtkCells.SetCells(points.shape[0], vtk_cell_data)


        self.vtkPoints.Modified()
        self.vtkCells.Modified()
        self.vtkPolyData.Modified()

    ##########################################################################
    def clear_points(self):
        self.vtkPoints = vtk.vtkPoints()
        self.vtkCells = vtk.vtkCellArray()
        self.vtkPolyData.SetPoints(self.vtkPoints)
        self.vtkPolyData.SetVerts(self.vtkCells)

##########################################################################
class PointSizeInteractorStyle(vtk.vtkInteractorStyleTrackballCamera):

    def __init__(self, parent, point_cloud=None, max_pt_size=20):
        self.AddObserver("MouseWheelForwardEvent", self.mouse_wheel_forward_event)
        self.AddObserver("MouseWheelBackwardEvent", self.mouse_wheel_backward_event)
        self.AddObserver("LeftButtonPressEvent", self.LeftButtonPressEvent)
        self._parent = parent
        self._point_cloud = point_cloud
        self._max_pt_size = max_pt_size

    ##########################################################################
    def mouse_wheel_forward_event(self, obj, event):
        if self._parent.GetAltKey():
            new_point_size = self._point_cloud_actor.GetProperty().GetPointSize() + 1
            new_point_size = self._max_pt_size if new_point_size > self._max_pt_size else new_point_size
            self._point_cloud.vtkActor.GetProperty().SetPointSize(new_point_size)
            self._parent.GetRenderWindow().Render()
        else:
            self.OnMouseWheelForward()
        return

    def LeftButtonPressEvent(self, obj, event):
        print("Global interactor")
        self.OnLeftButtonDown()

    ##########################################################################
    def mouse_wheel_backward_event(self, obj, event):
        if self._parent.GetAltKey():
            new_point_size = self._point_cloud_actor.GetProperty().GetPointSize() - 1
            new_point_size = 1 if  new_point_size < 1 else new_point_size
            self._point_cloud.vtkActor.GetProperty().SetPointSize(new_point_size)
            self._parent.GetRenderWindow().Render()
        else:
            self.OnMouseWheelBackward()
        return

##########################################################################
class PointCloudInteractorStyle(PointSizeInteractorStyle):

    def __init__(self, parent, image_actor, point_cloud, caption_renderer, pc_gen=None, max_pt_size=20):
        # self.AddObserver("KeyPressEvent", self.key_press_event)
        self._parent = parent
        self._image_actor = image_actor
        self._pc_gen = pc_gen
        self._caption_renderer = caption_renderer

        self._caption_renderer.RemoveAllViewProps()

        point_cloud.set_points(pts)
        super().__init__(parent, point_cloud)

#########################################################################
def main():
    point_cloud_cube = create_cylinder()
    # point_cloud_cube = point_cloud_cube + np.random.rand(*point_cloud_cube.shape) * 0.02
    R, _ = np.linalg.qr(np.random.rand(3, 3))
    #point_cloud_cube = transformation.rotate(R, point_cloud_cube.T).T
    point_cloud = np.vstack((point_cloud_cube))

    color = np.ones((point_cloud.shape[0], 3), dtype=point_cloud.dtype) * 255

    camera = vtk.vtkCamera()
    camera.SetViewUp(0, -1, 0)
    camera.SetPosition(2, -1, -4)
    camera.SetFocalPoint(0, 0, 0)
    point_cloud = np.hstack((point_cloud[:, :3], color))

    show_pointcloud(point_cloud, 'Test', camera=camera, axes_scaling=0.5)


if __name__ == '__main__':
    main()

Solution

  • I was able to solve it - therefore I had to postpone certain matrix manipulations by using oldTransform.PostMultiply() as you might see here...

    class ModBoxWidget(vtk.vtkBoxWidget):
    
        def __init__(self, fixed_Y_pos=None):
            self.AddObserver("InteractionEvent", self.horizontal_move_event)
            #super().__init__(self)
            self.fixed_Y_pos = fixed_Y_pos
    
        def horizontal_move_event(self, obj, event):
            oldTransform = vtk.vtkTransform()
            obj.GetTransform(oldTransform)
            q = np.zeros(4)
            oldTransform.GetOrientationWXYZ(q)
            oldTransform.PostMultiply()
    
            newTransform = vtk.vtkTransform()
            pos = np.zeros(3)
            polyData = vtk.vtkPolyData()
            obj.GetPolyData(polyData)
            num_points = polyData.GetNumberOfPoints()
            coords = np.zeros((num_points, 3))
            for i in range(num_points):
                polyData.GetPoint(i, coords[i])
    
            pos[0] = coords[14][0]
            pos[1] = coords[14][1]
            pos[2] = coords[14][2]
    
            newTransform.PostMultiply()
            newTransform.Identity()
            newTransform.Translate(-pos[0], -pos[1], -pos[2])
            print(f"{-q[0]*q[1]:3.2f} {-q[0]*q[1]:3.2f} {-q[0]*q[2]:3.2f}")
            newTransform.RotateX(-q[0]*q[1])
            newTransform.RotateY(-q[0]*q[2])
    
            newTransform.Translate(pos[0], pos[1], self.fixed_Y_pos or obj.GetProp3D().GetScale()[1]/2.0)
            oldTransform.Concatenate(newTransform)
            obj.SetTransform(oldTransform)
            obj.SetHandleSize(0.001)