Flandre923
1219 words
6 minutes
Untity学习记录-实现一个完整的2D游戏Demo-34-实现移动设备屏幕操作

在Unity中使用Input System实现跨平台的触控输入系统,让你的游戏能够完美适配移动设备。这个教程将带你从零开始创建虚拟摇杆和屏幕按钮,并实现平台自动检测功能。

1. 虚拟摇杆实现#

1.1 创建虚拟摇杆UI#

首先,我们需要在Canvas下创建虚拟摇杆的基础结构:


1. 在Main Canvas下创建空对象,命名为"MobileTouch"
2. 添加Image组件作为摇杆背景,命名为"StickBackground"
3. 添加Image组件作为摇杆手柄,命名为"StickHandle"
4. 调整Anchor Preset为左下角,方便移动设备操作
// 摇杆尺寸设置建议
stickBackground.rectTransform.sizeDelta = new Vector2(300, 300);
stickHandle.rectTransform.sizeDelta = new Vector2(150, 150);
stickBackground.color = new Color(1, 1, 1, 0.5f); // 半透明效果

1.2 添加On-Screen Stick组件#

使用Unity的Input System提供的OnScreen组件:

// 为摇杆背景添加OnScreenStick组件
var onScreenStick = stickBackground.gameObject.AddComponent<OnScreenStick>();

// 配置摇杆参数
onScreenStick.movementRange = 100f; // 摇杆移动范围
onScreenStick.controlPath = "<Gamepad>/leftStick"; // 映射到左摇杆

2. 虚拟按钮实现#

2.1 创建按钮布局#

在Canvas右侧创建四个虚拟按钮:

1. 在MobileTouch下创建空对象"RightPanel"
2. 添加四个Image作为按钮,分别命名为:
   - "ButtonNorth" (上/红色)
   - "ButtonEast" (右/绿色)
   - "ButtonSouth" (下/蓝色)
   - "ButtonWest" (左/黄色)
3. 使用Anchor Preset精确定位每个按钮

2.2 添加On-Screen Button组件#

// 为每个按钮添加OnScreenButton并映射到对应输入
void SetupVirtualButton(GameObject button, string controlPath)
{
    var onScreenBtn = button.AddComponent<OnScreenButton>();
    onScreenBtn.controlPath = controlPath;
}

// 按钮映射配置
SetupVirtualButton(buttonNorth, "<Gamepad>/buttonNorth"); // X/Y按钮
SetupVirtualButton(buttonEast, "<Gamepad>/buttonEast");   // B/圆形按钮
SetupVirtualButton(buttonSouth, "<Gamepad>/buttonSouth"); // A/交叉按钮
SetupVirtualButton(buttonWest, "<Gamepad>/buttonWest");   // Y/方形按钮

3. 平台自动检测与UI切换#

3.1 检测运行平台#

public class PlatformDetector : MonoBehaviour
{
    public GameObject mobileTouchUI;
  
    private void Awake()
    {
#if UNITY_STANDALONE
        // PC/Mac平台隐藏触控UI
        mobileTouchUI.SetActive(false);
#elif UNITY_IOS || UNITY_ANDROID
        // 移动平台显示触控UI
        mobileTouchUI.SetActive(true);
    
        // 移动平台特有设置
        Application.targetFrameRate = 60;
        Screen.orientation = ScreenOrientation.LandscapeLeft;
#endif
    }
}

3.2 优化移动设备输入#

// 为移动设备优化输入响应
public class MobileInputOptimizer : MonoBehaviour
{
    [SerializeField] private float stickDeadZone = 0.2f;
    [SerializeField] private float buttonPressScale = 0.9f;
  
    private void OnEnable()
    {
        var stick = GetComponent<OnScreenStick>();
        if(stick != null)
        {
            stick.deadZone = stickDeadZone;
        }
    }
  
    public void OnButtonPressed(Image buttonImage)
    {
        // 按钮按下效果
        buttonImage.transform.localScale = Vector3.one * buttonPressScale;
    }
  
    public void OnButtonReleased(Image buttonImage)
    {
        // 按钮释放效果
        buttonImage.transform.localScale = Vector3.one;
    }
}

4. Input System配置优化#

4.1 创建移动设备专用的Input Action Asset#

// 步骤:
1. 在Project窗口右键 -> Create -> Input Actions
2. 命名为"MobileInput"
3. 双击打开Input Action Editor
4. 添加"Touch" Action Map
5. 配置触控相关输入:
   - PrimaryTouch (触摸位置)
   - SecondaryTouch (双指操作)
   - Tap (点击)
   - Swipe (滑动)

4.2 在代码中处理触控输入#

public class MobileInputHandler : MonoBehaviour
{
    private InputAction primaryTouchAction;
  
    private void Awake()
    {
        var playerInput = GetComponent<PlayerInput>();
        primaryTouchAction = playerInput.actions["Touch/PrimaryTouch"];
    }
  
    private void OnEnable()
    {
        primaryTouchAction.performed += OnPrimaryTouch;
        primaryTouchAction.canceled += OnPrimaryTouchEnd;
    }
  
    private void OnPrimaryTouch(InputAction.CallbackContext context)
    {
        Vector2 touchPos = context.ReadValue<Vector2>();
        Debug.Log($"Primary touch at: {touchPos}");
    }
  
    private void OnPrimaryTouchEnd(InputAction.CallbackContext context)
    {
        Debug.Log("Primary touch ended");
    }
}

5. 多平台构建设置#

5.1 安装必要平台模块#

1. 打开Unity Hub
2. 选择当前项目使用的Unity版本
3. 点击右上角齿轮图标 -> Add Modules
4. 勾选需要添加的平台:
   - Android Build Support
   - iOS Build Support
   - WebGL Build Support
5. 点击Install完成安装

5.2 切换构建平台#

// 在代码中动态获取当前平台
RuntimePlatform platform = Application.platform;

// 平台特定逻辑
if(platform == RuntimePlatform.Android || platform == RuntimePlatform.IPhonePlayer)
{
    // 移动设备特有逻辑
}
else if(platform == RuntimePlatform.WebGLPlayer)
{
    // WebGL特有逻辑
}

6. 高级触控功能扩展#

6.1 实现手势识别#

public class GestureRecognizer : MonoBehaviour
{
    private Vector2 touchStartPos;
    private float touchStartTime;
  
    private void Update()
    {
        if(Input.touchCount > 0)
        {
            Touch touch = Input.GetTouch(0);
        
            switch(touch.phase)
            {
                case TouchPhase.Began:
                    touchStartPos = touch.position;
                    touchStartTime = Time.time;
                    break;
                
                case TouchPhase.Ended:
                    DetectGesture(touch.position);
                    break;
            }
        }
    }
  
    private void DetectGesture(Vector2 endPos)
    {
        float distance = Vector2.Distance(touchStartPos, endPos);
        float duration = Time.time - touchStartTime;
    
        if(distance > 100f && duration < 0.5f)
        {
            Vector2 direction = (endPos - touchStartPos).normalized;
        
            if(Mathf.Abs(direction.x) > Mathf.Abs(direction.y))
            {
                Debug.Log(direction.x > 0 ? "向右滑动" : "向左滑动");
            }
            else
            {
                Debug.Log(direction.y > 0 ? "向上滑动" : "向下滑动");
            }
        }
    }
}

6.2 虚拟摇杆动态调整#

public class DynamicJoystick : OnScreenStick
{
    [SerializeField] private float activeOpacity = 0.7f;
    [SerializeField] private float inactiveOpacity = 0.3f;
  
    private Image stickImage;
  
    protected override void Start()
    {
        base.Start();
        stickImage = GetComponent<Image>();
    }
  
    public override void OnPointerDown(PointerEventData eventData)
    {
        base.OnPointerDown(eventData);
        stickImage.color = new Color(1, 1, 1, activeOpacity);
    }
  
    public override void OnPointerUp(PointerEventData eventData)
    {
        base.OnPointerUp(eventData);
        stickImage.color = new Color(1, 1, 1, inactiveOpacity);
    }
}

7. 总结与最佳实践#

通过本教程,我们实现了:

  1. 完整的虚拟摇杆系统:支持8方向输入和力度控制
  2. 多功能虚拟按钮:映射到游戏手柄标准布局
  3. 智能平台检测:自动适配PC和移动设备
  4. 手势识别扩展:为高级操作提供可能

最佳实践建议

  1. 为不同屏幕尺寸设计响应式UI布局
  2. 添加触觉反馈增强移动设备体验
  3. 考虑左右手操作习惯提供自定义布局选项
  4. 为虚拟按钮添加按下/释放的视觉反馈
  5. 在Editor中模拟触控输入方便测试
Untity学习记录-实现一个完整的2D游戏Demo-34-实现移动设备屏幕操作
https://fuwari.vercel.app/posts/unity/2dadvanture/34mobiletouch/
Author
Flandre923
Published at
2025-03-31