1
votes

I have some camera with perspective projection and object in the (0,0,0). I computed "bounding frame"(size in pixels on screen) of this object according to this camera and scaled it to fit the screen size. And now it in the middle of screen.

Frame of the scnView is equal to the screen size and I have some UI components above this view(e.g. navigation bar and some big transparent view in the bottom of view). The scnView has some fullscreen background and this object can be scaled/moved/... therefore I need scnview to be fullscreen.

smth like that

Then I move this object in the middle of "clear area" (the area with absence of UIComponents) and see the bottom of this object (as if it's above me), because my projection is perspective.

I want to move this object in the center of clear area and see like it's in front of me without any distortion. How can I achieve that?

enter image description here

I see 2 solutions. 1) Draw my object offscreen in texture and after that draw this texture on screen in required position. 2) create scnView a bit taller. For example set frame size to = (0, (bottomHeight - topHeight) / 2, width, height + (bottomHeight - topHeight) / 2) to move the center of scnView in required position.

But I don't like first solution due to addition draw call and second solution even sounds crappy.

P.S. sorry for that pictures

Update: Basically I want to use perspective projection and move 3D object like a 2D image in SceneKit.

3

3 Answers

1
votes

Sorry if I'm not be understanding correctly, but can you strafe camera up and center it, or will that not be clear enough?

func strafeY(vAmount: XFloat)
{
  nPosition = cgVecAdd(v1: nPosition, v2: cgVecScalarMult(v: nTarget, s: vAmount))
}
1
votes

Try this... generate a new scenekit game project, then replace the default rotation with the code below.

Rotate X slightly as you move the object up to to "somewhat" maintain the relationship to the eye.

You could create a similar but separate routine to do smaller increments to smooth it out.

    ship.runAction(SCNAction.rotateBy(x: CGFloat(GLKMathDegreesToRadians(-90)), y: 0, z: 0, duration: 0))

    let vRotateAmount: Float = 3

    let vAction1 =  SCNAction.move(to: SCNVector3Make(0, 1, 0), duration: 1)
    let vAction1a = SCNAction.rotateBy(x: CGFloat(GLKMathDegreesToRadians(vRotateAmount)), y:0, z:0, duration: 1)

    let vAction2 = SCNAction.move(to: SCNVector3Make(0, 2, 0), duration: 1)
    let vAction2a = SCNAction.rotateBy(x: CGFloat(GLKMathDegreesToRadians(vRotateAmount)), y:0, z:0,  duration: 1)

    let vAction3 =  SCNAction.move(to: SCNVector3Make(0, 1, 0), duration: 1)
    let vAction3a = SCNAction.rotateBy(x: CGFloat(GLKMathDegreesToRadians(-vRotateAmount)), y:0, z:0,  duration: 1)

    let vAction4 = SCNAction.move(to: SCNVector3Make(0, 0, 0), duration: 1)
    let vAction4a = SCNAction.rotateBy(x: CGFloat(GLKMathDegreesToRadians(-vRotateAmount)), y:0, z:0,  duration: 1)

    let seq = SCNAction.sequence([vAction1, vAction1a, vAction2, vAction2a, vAction3, vAction3a, vAction4, vAction4a])
    let allSeq = SCNAction.repeatForever(seq)
    ship.runAction(allSeq)
0
votes

Solved it by changing camera projectionTransform. Smth like:

let newMatrix = SCNMatrix4Mult(initialProjectionCameraMatrix, translationMatrix)