Rotating around specific point

Hi all.

I have self.picker = vtk.vtkPointPicker() to select the point in the point cloud. And I would like to be able by double-click on some point, make this point center of the rotation WITHOUT shifting the camera so that the selected point is in the center of the viewport. I have the following function:

def _on_point_click(self, obj, event):
        click_position = self.interactor.GetEventPosition()
        self.picker.Pick(click_position[0], click_position[1], 0, self.renderer)
        if self.picker.GetPointId() != -1:
            coords = self.picker.GetPickPosition()        
            self.renderer.GetActiveCamera().SetFocalPoint(coords)

Of course, that one indeed allows to use the selected point as the rotation BUT at the same time all view is moved so that the new focal point is in the center of the view.

Is there way to avoid this or it is somehow against some basic VTK design?

I have seem this solution → c++ - How to rotate a vtk scene around a specific point? - Stack Overflow
But it is 8 years old and, well, honestly looks a bit of an overkill for a relative simple(?) feature.
Is this indeed the way I should I use, and just translate it to tools I am using (Python & pyside6)? Or there is some way to do it in the VTK itself? Thanks.

The second-ranked answer seems quite reasonable (in your SO link). Just set the focal point on the vtkCamera object. I haven’t tried though.

Hello,

It seems overkill but to rotate an object about a random position, you need, first to translate the object to the origin, apply the rotation and then translate it back to the former position. This is how computational geometry works.

best,

PC

As written in the comments to that answer as well, it also shifts the view so that the chosen point is in the center of the viewport.

Mathematically, indeed. But I was hoping for a more built-in support for this functionality since it is pretty common in multiple 3D viewers (CloudCompare, for example). But if there is nothing available in VTK itself, I guess the only way is to implement this workaround indeed.

You have to implement it by yourselfin your camera model. I post my java code doing this as an example:



 /**
     * Rotates the camera by mouse move delta values.
     * 
     * @param x           the delta
     * @param y           the delta
     * @param speedFactor
     */
    public void rotateCameraByDelta( int x, int y, float speedFactor ) {

        double scale = Norm( cam.GetPosition() );
        if ( scale <= 0.0 ) {
            scale = Norm( cam.GetFocalPoint() );
            if ( scale <= 0.0 ) {
                scale = 1.0;
            }
        }

        double[] temp = cam.GetFocalPoint();
        double[] fp   = cam.GetFocalPoint();
        cam.SetFocalPoint( temp[0] / scale, temp[1] / scale, temp[2] / scale );
        temp = cam.GetPosition();
        cam.SetPosition( temp[0] / scale, temp[1] / scale, temp[2] / scale );

        // translate to center
        rotTransform.Identity();
        rotTransform.Translate( COR.x / scale, COR.y / scale, COR.z / scale );

        // azimuth
        cam.OrthogonalizeViewUp();
        double[] viewUp = cam.GetViewUp();
        int[]    size   = renderer.GetSize();
        Vector3d up     = new Vector3d( viewUp[0], viewUp[1], viewUp[2] );
        rotTransform.RotateWXYZ( 360.0 * x * speedFactor / size[0], up.x, up.y, up.z );

        // elevation
        Vector3d v2      = new Vector3d();
        Vector3d projDir = new Vector3d( cam.GetDirectionOfProjection()[0], cam.GetDirectionOfProjection()[1], cam.GetDirectionOfProjection()[2] );

        v2.cross( projDir, up );
        rotTransform.RotateWXYZ( -360.0 * y * speedFactor / size[1], v2.x, v2.y, v2.z );

        // translate back
        rotTransform.Translate( -COR.x / scale, -COR.y / scale, -COR.z / scale );

        applyTransform( rotTransform );
        cam.OrthogonalizeViewUp();

        // For rescale back.
        temp = cam.GetFocalPoint();
        cam.SetFocalPoint( temp[0] * scale, temp[1] * scale, temp[2] * scale );
        temp = cam.GetPosition();
        cam.SetPosition( temp[0] * scale, temp[1] * scale, temp[2] * scale );

        double[] pos = new double[3];
        pos[0] = temp[0] * scale;
        pos[1] = temp[1] * scale;
        pos[2] = temp[2] * scale;

        viewportVTK.resetCameraClippingRange();
        if ( !isCam )
            Engine.INSTANCE.getViewportManager().updateCamera( pos, fp );
    }


applyTransfrom applies everything on the camera.

Thank you!

I just always try to check if something already exist hidden somewhere in the package, before I start implementing myself.