陀螺仪控制摄像头脚本 GyroController

using UnityEngine;

///

/// Gyroscope controller that works with any device orientation.

///

public class GyroController : MonoBehaviour

{

    #region [Private fields]

    private bool gyroEnabled = true;

    private const float lowPassFilterFactor = 0.2f;

    private readonly Quaternion baseIdentity = Quaternion.Euler(90, 0, 0);

    private readonly Quaternion landscapeRight = Quaternion.Euler(0, 0, 90);

    private readonly Quaternion landscapeLeft = Quaternion.Euler(0, 0, -90);

    private readonly Quaternion upsideDown = Quaternion.Euler(0, 0, 180);

    private Quaternion cameraBase = Quaternion.identity;

    private Quaternion calibration = Quaternion.identity;

    private Quaternion baseOrientation = Quaternion.Euler(90, 0, 0);

    private Quaternion baseOrientationRotationFix = Quaternion.identity;

    private Quaternion referanceRotation = Quaternion.identity;

    private bool debug = true;

    #endregion

    #region [Unity events]

    protected void Start()

    {

        AttachGyro();

    }

    protected void Update()

    {

        if (!gyroEnabled)

            return;

        transform.rotation = Quaternion.Slerp(transform.rotation,

            cameraBase * (ConvertRotation(referanceRotation * Input.gyro.attitude) * GetRotFix()), lowPassFilterFactor);

    }

    protected void OnGUI()

    {

        if (!debug)

            return;

        GUILayout.Label("Orientation: " + Screen.orientation);

        GUILayout.Label("Calibration: " + calibration);

        GUILayout.Label("Camera base: " + cameraBase);

        GUILayout.Label("input.gyro.attitude: " + Input.gyro.attitude);

        GUILayout.Label("transform.rotation: " + transform.rotation);

        if (GUILayout.Button("On/off gyro: " + Input.gyro.enabled, GUILayout.Height(100)))

        {

            Input.gyro.enabled = !Input.gyro.enabled;

        }

        if (GUILayout.Button("On/off gyro controller: " + gyroEnabled, GUILayout.Height(100)))

        {

            if (gyroEnabled)

            {

                DetachGyro();

            }

            else

            {

                AttachGyro();

            }

        }

        if (GUILayout.Button("Update gyro calibration (Horizontal only)", GUILayout.Height(80)))

        {

            UpdateCalibration(true);

        }

        if (GUILayout.Button("Update camera base rotation (Horizontal only)", GUILayout.Height(80)))

        {

            UpdateCameraBaseRotation(true);

        }

        if (GUILayout.Button("Reset base orientation", GUILayout.Height(80)))

        {

            ResetBaseOrientation();

        }

        if (GUILayout.Button("Reset camera rotation", GUILayout.Height(80)))

        {

            transform.rotation = Quaternion.identity;

        }

    }

    #endregion

    #region [Public methods]

    ///

    /// Attaches gyro controller to the transform.

    ///

    private void AttachGyro()

    {

        gyroEnabled = true;

        ResetBaseOrientation();

        UpdateCalibration(true);

        UpdateCameraBaseRotation(true);

        RecalculateReferenceRotation();

    }

    ///

    /// Detaches gyro controller from the transform

    ///

    private void DetachGyro()

    {

        gyroEnabled = false;

    }

    #endregion

    #region [Private methods]

    ///

    /// Update the gyro calibration.

    ///

    private void UpdateCalibration(bool onlyHorizontal)

    {

        if (onlyHorizontal)

        {

            var fw = (Input.gyro.attitude) * (-Vector3.forward);

            fw.z = 0;

            if (fw == Vector3.zero)

            {

                calibration = Quaternion.identity;

            }

            else

            {

                calibration = (Quaternion.FromToRotation(baseOrientationRotationFix * Vector3.up, fw));

            }

        }

        else

        {

            calibration = Input.gyro.attitude;

        }

    }

    ///

    /// Update the camera base rotation.

    ///

    ///

    /// Only y rotation.

    ///

    private void UpdateCameraBaseRotation(bool onlyHorizontal)

    {

        if (onlyHorizontal)

        {

            var fw = transform.forward;

            fw.y = 0;

            if (fw == Vector3.zero)

            {

                cameraBase = Quaternion.identity;

            }

            else

            {

                cameraBase = Quaternion.FromToRotation(Vector3.forward, fw);

            }

        }

        else

        {

            cameraBase = transform.rotation;

        }

    }

    ///

    /// Converts the rotation from right handed to left handed.

    ///

    ///

    /// The result rotation.

    ///

    ///

    /// The rotation to convert.

    ///

    private static Quaternion ConvertRotation(Quaternion q)

    {

        return new Quaternion(q.x, q.y, -q.z, -q.w);

    }

    ///

    /// Gets the rot fix for different orientations.

    ///

    ///

    /// The rot fix.

    ///

    private Quaternion GetRotFix()

    {

#if UNITY_3_5

        if (Screen.orientation == ScreenOrientation.Portrait)

            return Quaternion.identity;

        if (Screen.orientation == ScreenOrientation.LandscapeLeft || Screen.orientation == ScreenOrientation.Landscape)

            return landscapeLeft;    

        if (Screen.orientation == ScreenOrientation.LandscapeRight)

            return landscapeRight;

        if (Screen.orientation == ScreenOrientation.PortraitUpsideDown)

            return upsideDown;

        return Quaternion.identity;

#else

        return Quaternion.identity;

#endif

    }

    ///

    /// Recalculates reference system.

    ///

    private void ResetBaseOrientation()

    {

        baseOrientationRotationFix = GetRotFix();

        baseOrientation = baseOrientationRotationFix * baseIdentity;

    }

    ///

    /// Recalculates reference rotation.

    ///

    private void RecalculateReferenceRotation()

    {

        referanceRotation = Quaternion.Inverse(baseOrientation) * Quaternion.Inverse(calibration);

    }

    #endregion

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值