代码之家  ›  专栏  ›  技术社区  ›  Andy Jazz

SceneKit SCNMatrix4中元素的最低行是什么?

  •  0
  • Andy Jazz  · 技术社区  · 6 年前

    SCNMatrix4 结构,最后一个 m44

    m41 , m42 m43 SCNMatrix4中的元素?我到底能用这三个元素做什么?

    init(m11: Float, m12: Float, m13: Float, m14: Float, 
         m21: Float, m22: Float, m23: Float, m24: Float, 
         m31: Float, m32: Float, m33: Float, m34: Float, 
         m41: Float, m42: Float, m43: Float, m44: Float)
    

    enter image description here

    3 回复  |  直到 6 年前
        1
  •  2
  •   BlackMirrorz    6 年前

    我可能错了,但我想 m41 , m42 ,和 m43 result.worldTransform.columns.3 执行HIT测试时。

    在放置 SCNNode ARSCNHitTest 您可以使用:

    let hitTestTransform = SCNMatrix4(result.worldTransform)
    let positionFromMatrix4 = SCNVector3(hitTestTransform.m41, hitTestTransform.m42, hitTestTransform.m43)
    let positionFromColumns =  SCNVector3(result.worldTransform.columns.3.x, result.worldTransform.columns.3.y, result.worldTransform.columns.3.z) 
    

    下面的例子应该有助于澄清问题:

    /// Places Our Model At The Position Of An Existining ARPlaneAnchor
    ///
    /// - Parameter gesture: UITapGestureRecognizer
    @IBAction func placeModel(_ gesture: UITapGestureRecognizer){
    
        //1. Get The Touch Location
        let touchLocation = gesture.location(in: self.augmentedRealityView)
    
        //2. Perform An ARSCNHitTest For Any Existing Planes
        guard let result = self.augmentedRealityView.hitTest(touchLocation, types: [.existingPlane, .existingPlaneUsingExtent]).first else { return }
    
            //3. Get The World Transform
            let hitTestTransform = SCNMatrix4(result.worldTransform)
    
            //4. Initialize Our Position Either From .m41, .m42, .m43 Or From Columns.3
            let positionFromMatrix4 = SCNVector3(hitTestTransform.m41, hitTestTransform.m42, hitTestTransform.m43)
            let positionFromColumns =  SCNVector3(result.worldTransform.columns.3.x, result.worldTransform.columns.3.y, result.worldTransform.columns.3.z)
    
            //5. Log Them To Check I'm Not Being A Moron
            print(
                """
                Position From Matrix 4 == \(positionFromMatrix4)
                Position From Columns == \(positionFromColumns)
                """)
    
            /*
            Position From Matrix 4 == SCNVector3(x: -0.39050543, y: -0.004766479, z: 0.08107365)
            Position From Columns == SCNVector3(x: -0.39050543, y: -0.004766479, z: 0.08107365)
            */
    
            //6. Add A Node At The Position & Add It To The Hierachy
            let boxNode = SCNNode(geometry: SCNBox(width: 1, height: 1, length: 1, chamferRadius: 0))
            boxNode.geometry?.firstMaterial?.diffuse.contents = UIColor.cyan
            boxNode.position = positionFromMatrix4
            self.augmentedRealityView.scene.rootNode.addChildNode(boxNode)
    }
    

        2
  •  1
  •   Andy Jazz    6 年前

    基于@BlackMirrorz的答案。

    使用 m41 , m42 ,和 m43 元素 SCNMatrix4

        3
  •  1
  •   Rob Tow    6 年前

    最下面的一行用于平移和投影,例如透视图或相机观察截锥体。 https://referencesource.microsoft.com/#System.Numerics/System/Numerics/Matrix4x4.cs,78a107de58a17946 (从第864行开始),显示“[如何创建]基于视场、纵横比和远近视场平面距离的透视投影矩阵。”

        /// <summary>
        /// Creates a perspective projection matrix based on a field of view, aspect ratio, and near and far view plane distances. 
        /// </summary>
        /// <param name="fieldOfView">Field of view in the y direction, in radians.</param>
        /// <param name="aspectRatio">Aspect ratio, defined as view space width divided by height.</param>
        /// <param name="nearPlaneDistance">Distance to the near view plane.</param>
        /// <param name="farPlaneDistance">Distance to the far view plane.</param>
        /// <returns>The perspective projection matrix.</returns>
        public static Matrix4x4 CreatePerspectiveFieldOfView(float fieldOfView, float aspectRatio, float nearPlaneDistance, float farPlaneDistance)
        {
            if (fieldOfView <= 0.0f || fieldOfView >= Math.PI)
                throw new ArgumentOutOfRangeException("fieldOfView");
    
            if (nearPlaneDistance <= 0.0f)
                throw new ArgumentOutOfRangeException("nearPlaneDistance");
    
            if (farPlaneDistance <= 0.0f)
                throw new ArgumentOutOfRangeException("farPlaneDistance");
    
            if (nearPlaneDistance >= farPlaneDistance)
                throw new ArgumentOutOfRangeException("nearPlaneDistance");
    
            float yScale = 1.0f / (float)Math.Tan(fieldOfView * 0.5f);
            float xScale = yScale / aspectRatio;
    
            Matrix4x4 result;
    
            result.M11 = xScale;
            result.M12 = result.M13 = result.M14 = 0.0f;
    
            result.M22 = yScale;
            result.M21 = result.M23 = result.M24 = 0.0f;
    
            result.M31 = result.M32 = 0.0f;
            result.M33 = farPlaneDistance / (nearPlaneDistance - farPlaneDistance);
            result.M34 = -1.0f;
    
            result.M41 = result.M42 = result.M44 = 0.0f;
            result.M43 = nearPlaneDistance * farPlaneDistance / (nearPlaneDistance - farPlaneDistance);
    
            return result;
        }
    

    阿基特-它得到摄像机的位置。。。 https://gist.github.com/fisherds/a2a75c914c293213f594daf5fd940d3d

    @IBAction func pressedAddEarth(_ sender: Any) {
    
        guard let pointOfView = sceneView.pointOfView else {return}
        let transform = pointOfView.transform
        let orientation = SCNVector3(-transform.m31, -transform.m32, -transform.m33)
        let location = SCNVector3(transform.m41, transform.m42, transform.m43)
        let currentPositionOfCamera = orientation + location
    
        // other stuff   
    }    
    
    func +(left: SCNVector3, right: SCNVector3) -> SCNVector3 {
      return SCNVector3Make(left.x + right.x, left.y + right.y, left.z + right.z)
    }