unity 摇杆

/***********************************************
                EasyTouch Controls
    Copyright © 2016 The Hedgehog Team
      http://www.thehedgehogteam.com/Forum/
        
      The.Hedgehog.Team@gmail.com
        
**********************************************/
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
using UnityEngine.Events;
using UnityEngine.EventSystems;

[System.Serializable]
public class ETCJoystick : ETCBase,IPointerEnterHandler,IDragHandler, IBeginDragHandler, IPointerDownHandler, IPointerUpHandler {
        
    #region Unity Events
    [System.Serializable] public class OnMoveStartHandler : UnityEvent{}
    [System.Serializable] public class OnMoveSpeedHandler : UnityEvent<Vector2> { }
    [System.Serializable] public class OnMoveHandler : UnityEvent<Vector2> { }
    [System.Serializable] public class OnMoveEndHandler : UnityEvent{ }

    [System.Serializable] public class OnTouchStartHandler : UnityEvent{}
    [System.Serializable] public class OnTouchUpHandler : UnityEvent{ }

    [System.Serializable] public class OnDownUpHandler : UnityEvent{ }
    [System.Serializable] public class OnDownDownHandler : UnityEvent{ }
    [System.Serializable] public class OnDownLeftHandler : UnityEvent{ }
    [System.Serializable] public class OnDownRightHandler : UnityEvent{ }

    [System.Serializable] public class OnPressUpHandler : UnityEvent{ }
    [System.Serializable] public class OnPressDownHandler : UnityEvent{ }
    [System.Serializable] public class OnPressLeftHandler : UnityEvent{ }
    [System.Serializable] public class OnPressRightHandler : UnityEvent{ }


    [SerializeField] public OnMoveStartHandler onMoveStart;
    [SerializeField] public OnMoveHandler onMove;
    [SerializeField] public OnMoveSpeedHandler onMoveSpeed;
    [SerializeField] public OnMoveEndHandler onMoveEnd;

    [SerializeField] public OnTouchStartHandler onTouchStart;
    [SerializeField] public OnTouchUpHandler onTouchUp;

    [SerializeField] public OnDownUpHandler OnDownUp;
    [SerializeField] public OnDownDownHandler OnDownDown;
    [SerializeField] public OnDownLeftHandler OnDownLeft;
    [SerializeField] public OnDownRightHandler OnDownRight;

    [SerializeField] public OnDownUpHandler OnPressUp;
    [SerializeField] public OnDownDownHandler OnPressDown;
    [SerializeField] public OnDownLeftHandler OnPressLeft;
    [SerializeField] public OnDownRightHandler OnPressRight;
    #endregion

    #region Enumeration
    public enum JoystickArea { UserDefined,FullScreen, Left,Right,Top, Bottom, TopLeft, TopRight, BottomLeft, BottomRight};
    public enum JoystickType {Dynamic, Static};
    public enum RadiusBase {Width, Height, UserDefined};
    #endregion

    #region Members

    #region Public members
    public JoystickType joystickType;
    public bool allowJoystickOverTouchPad;
    public RadiusBase radiusBase;
    public float radiusBaseValue;
    public ETCAxis axisX;
    public ETCAxis axisY;
    public RectTransform thumb;
    
    public JoystickArea joystickArea;
    public RectTransform userArea;

    public bool isTurnAndMove = false;
    public float tmSpeed = 10;
    public float tmAdditionnalRotation = 0;
    public AnimationCurve tmMoveCurve;
    public bool tmLockInJump = false;
    private Vector3 tmLastMove;

    #endregion
        
    #region Private members
    private Vector2 thumbPosition;
    private bool isDynamicActif;
    private Vector2 tmpAxis;
    private Vector2 OldTmpAxis;
    private bool isOnTouch;


    #endregion

    #region Joystick behavior option
    [SerializeField]
    private bool isNoReturnThumb;
    public bool IsNoReturnThumb {
        get {
            return isNoReturnThumb;
        }
        set {
            isNoReturnThumb = value;
        }
    }    

    private Vector2 noReturnPosition;
    private Vector2 noReturnOffset;

    [SerializeField]
    private bool isNoOffsetThumb;
    public bool IsNoOffsetThumb {
        get {
            return isNoOffsetThumb;
        }
        set {
            isNoOffsetThumb = value;
        }
    }
    #endregion

    #region Inspector


    #endregion

    #endregion
    
    #region Constructor
    public ETCJoystick(){
        joystickType = JoystickType.Static;
        allowJoystickOverTouchPad = false;
        radiusBase = RadiusBase.Width;

        axisX = new ETCAxis("Horizontal");
        axisY = new ETCAxis("Vertical");
    
        _visible = true;
        _activated = true;

        joystickArea = JoystickArea.FullScreen;

        isDynamicActif = false;
        isOnDrag = false;
        isOnTouch = false;

        axisX.unityAxis = "Horizontal";
        axisY.unityAxis = "Vertical";

        enableKeySimulation = true;

        isNoReturnThumb = false;

        showPSInspector = false;
        showAxesInspector = false;
        showEventInspector = false;
        showSpriteInspector = false;
    }
    #endregion

    #region Monobehaviours Callback
    protected override void Awake (){

        base.Awake ();

        if (joystickType == JoystickType.Dynamic){
            this.rectTransform().anchorMin = new Vector2(0.5f,0.5f);
            this.rectTransform().anchorMax = new Vector2(0.5f,0.5f);
            this.rectTransform().SetAsLastSibling();
            visible = false;
        }

        if (allowSimulationStandalone && enableKeySimulation && !Application.isEditor && joystickType!=JoystickType.Dynamic){
            SetVisible(visibleOnStandalone);
        }
    }

    public override void Start(){
    
        axisX.InitAxis();
        axisY.InitAxis();

        if (enableCamera){
            InitCameraLookAt();
        }

        tmpAxis = Vector2.zero;
        OldTmpAxis = Vector2.zero;

        noReturnPosition = thumb.position;

        pointId = -1;

        if (joystickType == JoystickType.Dynamic){
            visible = false;
        }

        base.Start();

        // Init Camera position
        if (enableCamera && cameraMode == CameraMode.SmoothFollow){
            if (cameraTransform && cameraLookAt){
                cameraTransform.position = cameraLookAt.TransformPoint( new Vector3(0,followHeight,-followDistance));
                cameraTransform.LookAt( cameraLookAt);
            }
        }

        if (enableCamera && cameraMode == CameraMode.Follow){
            if (cameraTransform && cameraLookAt){
                cameraTransform.position = cameraLookAt.position + followOffset;
                cameraTransform.LookAt( cameraLookAt.position);
            }
        }
    }


    public override void Update (){

        base.Update ();

        #region dynamic joystick
        if (joystickType == JoystickType.Dynamic && !_visible && _activated){
            Vector2 localPosition = Vector2.zero;
            Vector2 screenPosition = Vector2.zero;
            
            if (isTouchOverJoystickArea(ref localPosition, ref screenPosition)){
                
                GameObject overGO = GetFirstUIElement( screenPosition);
                
                if (overGO == null || (allowJoystickOverTouchPad && overGO.GetComponent<ETCTouchPad>()) || (overGO != null && overGO.GetComponent<ETCArea>() ) ) {

                    cachedRectTransform.anchoredPosition = localPosition;
                    visible = true;
                }
            }
        }


        if (joystickType == JoystickType.Dynamic && _visible){
            if (GetTouchCount()==0){
                visible = false;

            }
        }
        #endregion

    }

    public override void LateUpdate (){

        if (enableCamera && !cameraLookAt ){
            InitCameraLookAt();
        }
        base.LateUpdate ();

    }

    private void InitCameraLookAt(){

        if (cameraTargetMode == CameraTargetMode.FromDirectActionAxisX){
            cameraLookAt = axisX.directTransform;
        }
        else if (cameraTargetMode == CameraTargetMode.FromDirectActionAxisY){
            cameraLookAt = axisY.directTransform;
            if (isTurnAndMove){
                cameraLookAt = axisX.directTransform;
            }
        }
        else if (cameraTargetMode == CameraTargetMode.LinkOnTag){
            GameObject tmpobj = GameObject.FindGameObjectWithTag(camTargetTag);
            if (tmpobj){
                cameraLookAt = tmpobj.transform;
            }
        }

        if (cameraLookAt)
            cameraLookAtCC = cameraLookAt.GetComponent<CharacterController>();
    }

    protected override void UpdateControlState (){
    
        if (_visible){
            UpdateJoystick();
        }
        else{
            if (joystickType == JoystickType.Dynamic){
                OnUp( false);
            }
        }
    }

    #endregion
    
    #region UI Callback
    public void OnPointerEnter(PointerEventData eventData){

        if (joystickType == JoystickType.Dynamic && !isDynamicActif && _activated &&  pointId==-1){
            eventData.pointerDrag = gameObject;
            eventData.pointerPress = gameObject;

            isDynamicActif = true;
            pointId = eventData.pointerId;
        }

        if (joystickType == JoystickType.Dynamic &&  !eventData.eligibleForClick){
            OnPointerUp( eventData );
        }

    }

    public void OnPointerDown(PointerEventData eventData){
        onTouchStart.Invoke();
        pointId = eventData.pointerId;
        if (isNoOffsetThumb){
            OnDrag( eventData);
        }
    }

    public void OnBeginDrag(PointerEventData eventData){


    }
    
    public void OnDrag(PointerEventData eventData){

        if (pointId == eventData.pointerId){
            isOnDrag = true;
            isOnTouch = true;

            float radius =  GetRadius();


            if (!isNoReturnThumb){
                thumbPosition =  (eventData.position - eventData.pressPosition);// / (cachedRootCanvas.rectTransform().localScale.x  ) ;
            }
            else{
                thumbPosition =((eventData.position - noReturnPosition) /cachedRootCanvas.rectTransform().localScale.x) + noReturnOffset;
            }

            if (isNoOffsetThumb){
                thumbPosition =  (eventData.position - (Vector2)cachedRectTransform.position) / cachedRootCanvas.rectTransform().localScale.x;
            }

            thumbPosition.x = Mathf.FloorToInt( thumbPosition.x);
            thumbPosition.y = Mathf.FloorToInt( thumbPosition.y);


            if (!axisX.enable){
                thumbPosition.x=0;
            }

            if (!axisY.enable){
                thumbPosition.y=0;
            }

            if (thumbPosition.magnitude > radius){
                if (!isNoReturnThumb){
                    thumbPosition = thumbPosition.normalized * radius ;
                }
                else{
                    thumbPosition = thumbPosition.normalized * radius;
                }
            }

            thumb.anchoredPosition =  thumbPosition; 
        }
    }

    public void OnPointerUp (PointerEventData eventData){
        if (pointId == eventData.pointerId){
            OnUp();
        }
    }

    private void OnUp(bool real=true){

        isOnDrag = false;
        isOnTouch = false;
        
        if (isNoReturnThumb){
            noReturnPosition = thumb.position;
            noReturnOffset = thumbPosition;
        }
        
        if (!isNoReturnThumb){
            thumbPosition =  Vector2.zero;
            thumb.anchoredPosition = Vector2.zero;
            
            axisX.axisState = ETCAxis.AxisState.None;
            axisY.axisState = ETCAxis.AxisState.None;
        }
        
        if (!axisX.isEnertia && !axisY.isEnertia){
            axisX.ResetAxis();
            axisY.ResetAxis();
            tmpAxis = Vector2.zero;
            OldTmpAxis = Vector2.zero;
            if (real){
                onMoveEnd.Invoke();
            }
        }
        
        if (joystickType == JoystickType.Dynamic){
            visible = false;
            isDynamicActif = false;
        }

        pointId=-1;

        if (real){
            onTouchUp.Invoke();
        }
    }
    #endregion

    #region Joystick Update
    protected override void DoActionBeforeEndOfFrame (){
        axisX.DoGravity();
        axisY.DoGravity();
    }

    private void UpdateJoystick(){

        #region Unity axes
        //enableKeySimulation && 
        if (!isOnTouch && _activated && _visible ){

            float x = Input.GetAxis(axisX.unityAxis);
            float y= Input.GetAxis(axisY.unityAxis);

            if (!isNoReturnThumb){
                thumb.localPosition = Vector2.zero;
            }

            isOnDrag = false;

            if (x!=0){
                isOnDrag = true;
                thumb.localPosition = new Vector2(GetRadius()*x, thumb.localPosition.y);
            }

            if (y!=0){
                isOnDrag = true;
                thumb.localPosition = new Vector2(thumb.localPosition.x,GetRadius()*y );
            }

            thumbPosition = thumb.localPosition;
        }
        #endregion

        // Computejoystick value
        OldTmpAxis.x = axisX.axisValue;
        OldTmpAxis.y = axisY.axisValue;

        tmpAxis = thumbPosition / GetRadius();

        axisX.UpdateAxis( tmpAxis.x,isOnDrag, ETCBase.ControlType.Joystick,true);
        axisY.UpdateAxis( tmpAxis.y,isOnDrag, ETCBase.ControlType.Joystick,true);

        #region Move event
        if ((axisX.axisValue!=0 ||  axisY.axisValue!=0 ) && OldTmpAxis == Vector2.zero){
            onMoveStart.Invoke();
        }
        if (axisX.axisValue!=0 ||  axisY.axisValue!=0 ){

            if (!isTurnAndMove){
                // X axis
                if( axisX.actionOn == ETCAxis.ActionOn.Down && (axisX.axisState == ETCAxis.AxisState.DownLeft || axisX.axisState == ETCAxis.AxisState.DownRight)){
                    axisX.DoDirectAction();
                }
                else if (axisX.actionOn == ETCAxis.ActionOn.Press){
                    axisX.DoDirectAction();
                }

                // Y axis
                if( axisY.actionOn == ETCAxis.ActionOn.Down && (axisY.axisState == ETCAxis.AxisState.DownUp || axisY.axisState == ETCAxis.AxisState.DownDown)){
                    axisY.DoDirectAction();
                }
                else if (axisY.actionOn == ETCAxis.ActionOn.Press){
                    axisY.DoDirectAction();
                }
            }
            else{
                DoTurnAndMove();
            }
            onMove.Invoke( new Vector2(axisX.axisValue,axisY.axisValue));
            onMoveSpeed.Invoke( new Vector2(axisX.axisSpeedValue,axisY.axisSpeedValue));
        }
        else if (axisX.axisValue==0 &&  axisY.axisValue==0  && OldTmpAxis!=Vector2.zero) {
            onMoveEnd.Invoke();
        }        

        if (!isTurnAndMove){
            if (axisX.axisValue==0 &&  axisX.directCharacterController ){
                if (!axisX.directCharacterController.isGrounded && axisX.isLockinJump)
                    axisX.DoDirectAction();
            } 

            if (axisY.axisValue==0 &&  axisY.directCharacterController ){
                if (!axisY.directCharacterController.isGrounded && axisY.isLockinJump)
                    axisY.DoDirectAction();
            }
        }
        else{
            if ((axisX.axisValue==0 && axisY.axisValue==0) &&  axisX.directCharacterController ){
                if (!axisX.directCharacterController.isGrounded && tmLockInJump)
                    DoTurnAndMove();
            }
        }

        #endregion


        #region Down & press event
        float coef =1;
        if (axisX.invertedAxis) coef = -1;
        if (Mathf.Abs(OldTmpAxis.x)< axisX.axisThreshold && Mathf.Abs(axisX.axisValue)>=axisX.axisThreshold){

            if (axisX.axisValue*coef >0){
                axisX.axisState = ETCAxis.AxisState.DownRight;
                OnDownRight.Invoke();
            }
            else if (axisX.axisValue*coef<0){
                axisX.axisState = ETCAxis.AxisState.DownLeft;
                OnDownLeft.Invoke();
            }
            else{
                axisX.axisState = ETCAxis.AxisState.None;
            }
        }
        else if (axisX.axisState!= ETCAxis.AxisState.None) {
            if (axisX.axisValue*coef>0){
                axisX.axisState = ETCAxis.AxisState.PressRight;
                OnPressRight.Invoke();
            }
            else if (axisX.axisValue*coef<0){
                axisX.axisState = ETCAxis.AxisState.PressLeft;
                OnPressLeft.Invoke();
            }
            else{
                axisX.axisState = ETCAxis.AxisState.None;
            }
        }

        coef =1;
        if (axisY.invertedAxis) coef = -1;
        if (Mathf.Abs(OldTmpAxis.y)< axisY.axisThreshold && Mathf.Abs(axisY.axisValue)>=axisY.axisThreshold  ){
            
            if (axisY.axisValue*coef>0){
                axisY.axisState = ETCAxis.AxisState.DownUp;
                OnDownUp.Invoke();
            }
            else if (axisY.axisValue*coef<0){
                axisY.axisState = ETCAxis.AxisState.DownDown;
                OnDownDown.Invoke();
            }
            else{
                axisY.axisState = ETCAxis.AxisState.None;
            }
        }
        else if (axisY.axisState!= ETCAxis.AxisState.None) {
            if (axisY.axisValue*coef>0){
                axisY.axisState = ETCAxis.AxisState.PressUp;
                OnPressUp.Invoke();
            }
            else if (axisY.axisValue*coef<0){
                axisY.axisState = ETCAxis.AxisState.PressDown;
                OnPressDown.Invoke();
            }
            else{
                axisY.axisState = ETCAxis.AxisState.None;
            }
        }
        #endregion

    }        
    #endregion

    #region Touch manager
    private bool isTouchOverJoystickArea(ref Vector2 localPosition, ref Vector2 screenPosition){
        
        bool touchOverArea = false;
        bool doTest = false;
        screenPosition = Vector2.zero;
        
        int count = GetTouchCount();
        int i=0;
        while (i<count && !touchOverArea){
            #if ((UNITY_ANDROID || UNITY_IOS || UNITY_WINRT || UNITY_BLACKBERRY) && !UNITY_EDITOR) 
            if (Input.GetTouch(i).phase == TouchPhase.Began){
                screenPosition = Input.GetTouch(i).position;
                doTest = true;
            }
            #else
            if (Input.GetMouseButtonDown(0)){
                screenPosition = Input.mousePosition;
                doTest = true;

            }
            #endif
            
            if (doTest && isScreenPointOverArea(screenPosition, ref localPosition) ){
                touchOverArea = true;
            }
            
            i++;
        }
        
        return touchOverArea;
    }
    
    private bool isScreenPointOverArea(Vector2 screenPosition, ref Vector2 localPosition){
        
        bool returnValue = false;
        
        if (joystickArea != JoystickArea.UserDefined){
            if (RectTransformUtility.ScreenPointToLocalPointInRectangle( cachedRootCanvas.rectTransform(),screenPosition,null,out localPosition)){
                
                switch (joystickArea){
                case JoystickArea.Left:
                    if (localPosition.x<0){
                        returnValue=true;
                    }
                    break;
                    
                case JoystickArea.Right:
                    if (localPosition.x>0){
                        returnValue = true;
                    }
                    break;
                    
                case JoystickArea.FullScreen:
                    returnValue = true;
                    break;
                    
                case JoystickArea.TopLeft:
                    if (localPosition.y>0 && localPosition.x<0){
                        returnValue = true;
                    }
                    break;
                case JoystickArea.Top:
                    if (localPosition.y>0){
                        returnValue = true;
                    }
                    break;
                    
                case JoystickArea.TopRight:
                    if (localPosition.y>0 && localPosition.x>0){
                        returnValue=true;
                    }
                    break;
                    
                case JoystickArea.BottomLeft:
                    if (localPosition.y<0 && localPosition.x<0){
                        returnValue = true;
                    }
                    break;
                    
                case JoystickArea.Bottom:
                    if (localPosition.y<0){
                        returnValue = true;
                    }
                    break;
                    
                case JoystickArea.BottomRight:
                    if (localPosition.y<0 && localPosition.x>0){
                        returnValue = true;
                    }
                    break;
                }
            }
        }
        else{
            if (RectTransformUtility.RectangleContainsScreenPoint( userArea, screenPosition, cachedRootCanvas.worldCamera )){
                RectTransformUtility.ScreenPointToLocalPointInRectangle( cachedRootCanvas.rectTransform(),screenPosition,cachedRootCanvas.worldCamera,out localPosition);
                returnValue = true;
            }
        }
        
        return returnValue;
        
    }
    
    private int GetTouchCount(){
        #if ((UNITY_ANDROID || UNITY_IOS || UNITY_WINRT || UNITY_BLACKBERRY) && !UNITY_EDITOR) 
        return Input.touchCount;
        #else
        if (Input.GetMouseButton(0) || Input.GetMouseButtonUp(0)){
            return 1;
        }
        else{
            return 0;
        }
        #endif
    }
    #endregion

    #region Other private method
    public float GetRadius(){

        float radius =0;
        
        switch (radiusBase){
        case RadiusBase.Width:
            radius = cachedRectTransform.sizeDelta.x * 0.5f;
            break;
        case RadiusBase.Height:
            radius = cachedRectTransform.sizeDelta.y * 0.5f;
            break;
        case RadiusBase.UserDefined:
            radius = radiusBaseValue;
            break;
        }
        
        return radius;
    }

    protected override void SetActivated (){
    
        GetComponent<CanvasGroup>().blocksRaycasts = _activated;

        if (!_activated){
            OnUp(false);
        }
    }

    protected override void SetVisible (bool visible=true){

        bool localVisible = _visible;
        if (!visible){
            localVisible = visible;
        }
        GetComponent<Image>().enabled = localVisible;
        thumb.GetComponent<Image>().enabled = localVisible;
        GetComponent<CanvasGroup>().blocksRaycasts = _activated;


    }
    #endregion


    private void DoTurnAndMove(){

        float angle =Mathf.Atan2( axisX.axisValue,axisY.axisValue ) * Mathf.Rad2Deg;
        float speed = tmMoveCurve.Evaluate( new Vector2(axisX.axisValue,axisY.axisValue).magnitude) * tmSpeed;

        if (axisX.directTransform != null){

            axisX.directTransform.rotation = Quaternion.Euler(new Vector3(0,  angle + tmAdditionnalRotation,0));

            if (axisX.directCharacterController != null){
                if (axisX.directCharacterController.isGrounded || !tmLockInJump){
                    Vector3 move = axisX.directCharacterController.transform.TransformDirection(Vector3.forward) *  speed;
                    axisX.directCharacterController.Move(move* Time.deltaTime);
                    tmLastMove = move;
                }
                else{
                    axisX.directCharacterController.Move(tmLastMove* Time.deltaTime);
                }
            }
            else{
                axisX.directTransform.Translate(Vector3.forward *  speed * Time.deltaTime,Space.Self);
            }
        }

    }

    public void InitCurve(){
        axisX.InitDeadCurve();
        axisY.InitDeadCurve();
        InitTurnMoveCurve();
    }

    public void InitTurnMoveCurve(){
        tmMoveCurve = AnimationCurve.EaseInOut(0,0,1,1);
        tmMoveCurve.postWrapMode = WrapMode.PingPong;
        tmMoveCurve.preWrapMode = WrapMode.PingPong;
    }
}

 

posted on 2024-10-08 17:38  zqiang0803  阅读(9)  评论(0编辑  收藏  举报

导航