You don't need this much math — it's all math that SceneKit is effectively already doing, so you just need to ask it for the right results.
[How do I] produce the SCNVector3 that puts the node in "front" of the camera at a given distance?
In the local coordinate system of a node hosting a camera, "front" is the -Z direction. If you put something at (0, 0, -10)
relative to the camera, it's directly centered in the camera's line of sight and 10 units away. There's two ways to do that:
Just make the text node a child node of the camera.
cameraNode.addChildNode(textNode)
textNode.position = SCNVector3(x: 0, y: 0, z: -10)
Note that this also makes it move with the camera: if the camera changes orientation or position, textNode
comes along for the ride. (Imagine a selfie stick in reverse — pivot the phone at one end, and the other end swings around with it.)
Since the text node is in camera-node space, you probably don't need anything more to make it face the camera, either. (The default orientation of a SCNText
relative to its parent space makes the text face the camera.)
(New in iOS 11 / macOS 10.13 / Xcode 9 / etc.) SceneKit now includes some convenience accessors and methods for doing various relative-geometry math in fewer steps. Most of those properties and methods are available in new flavors that support the system-wide SIMD library instead of using SCN vector/matrix types — and the SIMD library has nice features like overloaded operators built in. Putting that together, operations like this become a one-liner:
textNode.simdPosition = camera.simdWorldFront * distance
(simdWorldFront
is a vector of length 1.0, so you can change the distance just by scaling it.)
If you're still supporting iOS 10 / macOS 10.12 / etc, you can still use the node hierarchy to do coordinate conversions. The text node can be a child of the scene (the scene's rootNode
, actually), but you can find the scene-coordinate-space position corresponding to the camera-space coordinates you want. You don't really need a function for it, but this might do the trick:
func sceneSpacePosition(inFrontOf node: SCNNode, atDistance distance: Float) -> SCNVector3 {
let localPosition = SCNVector3(x: 0, y: 0, z: CGFloat(-distance))
let scenePosition = node.convertPosition(localPosition, to: nil)
// to: nil is automatically scene space
return scenePosition
}
textNode.position = sceneSpacePosition(inFrontOf: camera, atDistance: 3)
In this case, the text node is initially positioned relative to the camera, but doesn't stay "connected" to it. If you move the camera, the node stays where it is. Since the text node is in scene space, its orientation isn't connected to the camera, either.
[How do I] produce the SCNVector4 (or SCNQuaternion) that orients the node so it is "facing" the camera?
This part you either don't need to do much for or can hand over entirely to SceneKit, depending on which answer to the first question you used.
If you did (1) above — making the text node a child of the camera node — making it face the camera is a one-step thing. And as I noted, with SCNText
as your test geometry that step is already done. In its local coordinate space, an SCNText
is upright and readable from the default camera orientation (that is, looking in the -Z direction, with +Y up and +X to the right).
If you have some other geometry you want to face the camera, you need orient it only once, but you'll have to work out the appropriate orientation. For example, an SCNPyramid
has its point in the +Y direction, so if you want the point to face toward the camera (watch out, it's sharp), you'll want to rotate it around its X-axis by .pi/2
. (You can do this with eulerAngles
, rotation
, orientation
or pivot
.)
Since it's a child of the camera, any rotation you give it will be relative to the camera, so if the camera moves it'll stay pointed at the camera.
If you did (2) above, you could try to work out in scene-space terms what rotation is needed to face the camera, and apply that whenever the node or the camera move. But there's an API to do that for you. Actually, two APIs: SCNLookAtConstraint
makes any node "face towards" any other node (as determined by the -Z direction of the constrained node), and SCNBillboardConstraint
is sort of the same thing but with some extra considerations for the assumption that you want to face not just some other node, but specifically the camera.
It works pretty well just with default options:
textNode.constraints = [ SCNBillboardConstraint() ]
An extra caveat
If you're using the sceneSpacePosition
function above, and your camera node's orientation is set at render time — that is, you don't set transform
, eulerAngles
, rotation
, or orientation
on it directly, but instead use constraints, actions, animations, or physics to move/orient the camera — there's one more thing to consider.
SceneKit actually has two node hierarchies in play at any given time: the "model" tree and the "presentation" tree. The model tree is what directly setting transform
(etc) does. All those other ways of moving/orienting a node use the presentation tree. So, if you want to calculate positions using camera-relative space, you'll need the camera's presentation node:
textNode.position = sceneSpacePosition(inFrontOf: camera.presentation, atDistance: 3)
~~~~~~~~~~~~~
(Why is it this way? Among other things, it's so you can have implicit animation: set position
inside a transaction, and it'll animate from the old position to the new. During that animation, the model position
is what you set it to, and the presentation node is continuously changing its position
.)
与恶龙缠斗过久,自身亦成为恶龙;凝视深渊过久,深渊将回以凝视…