Allen Kashiwa Blog

羁旅红尘不曾醉,笑歌年少且长行。

专注游戏开发与设计


唯佳人与游戏不可辜负

眼见为实——如何在Unity中画出碰撞范围

碰撞器

游戏中经常需要做范围判定,常见的方式是进行物理上的碰撞检测。在Unity引擎中提供了各种类型的Collider和碰撞检测接口。其中,Collider可以在Scene窗口中看到绿色线标识的范围。例如,CapsuleCollider:

CapsuleCollider

这样的范围标识能带来直观的感受。但是在开发过程中,范围判定不一定全是通过Collider来做检测,并且挂载Collider的物体极有可能在我们需要看到显示范围时已经被销毁。所以,我们有必要做一个能画出各种几何图形的工具。Unity中提供的GizmosDraw的功能可以实现这样一个工具。

效果

我们的工具要能做到以下效果:

GizmosResult

从图中可以看出,GameObjective和GizmosDraw出的范围有出入,这是模拟Collider会随着设置的Scale,Radius,Center等值而产生偏差。而我们的工具需要保证在参数设置相同的情况下,Gizmos的结果和碰撞器的真实范围一致。

矩形

球形

胶囊体

代码

由于篇幅原因,这里仅给出DrawCircleImp和DrawCapsuleImp的实现代码。完整代码欢迎大家前往我的github围观。有任何问题也可以和我交流。另外,从本项目起,Untiy版本更新至5.6.0f3。

private static void DrawCircleImp(Vector3 center, Vector3 up, Color color, float radius)
{
    var oldColor = Gizmos.color;
    Gizmos.color = color;

    up = (up == Vector3.zero ? Vector3.up : up).normalized * radius;
    var forward = Vector3.Slerp(up, -up, 0.5f);
    var right = Vector3.Cross(up, forward).normalized * radius;
    for (var i = 1; i < 26; i++)
    {
        Gizmos.DrawLine(center + Vector3.Slerp(forward, right, (i - 1) / 25f), center + Vector3.Slerp(forward, right, i / 25f));
        Gizmos.DrawLine(center + Vector3.Slerp(forward, -right, (i - 1) / 25f), center + Vector3.Slerp(forward, -right, i / 25f));
        Gizmos.DrawLine(center + Vector3.Slerp(right, -forward, (i - 1) / 25f), center + Vector3.Slerp(right, -forward, i / 25f));
        Gizmos.DrawLine(center + Vector3.Slerp(-right, -forward, (i - 1) / 25f), center + Vector3.Slerp(-right, -forward, i / 25f));
    }

    Gizmos.color = oldColor;
}
        
private void DrawCapsuleImp(Vector3 pos, Vector3 center, Vector3 scale, CapsuleDirection direction, float radius, float height, Color color)
{
    // 参数保护
    if (height < 0f)
    {
        Debug.LogWarning("Capsule height can not be negative!");
        return;
    }
    if (radius < 0f)
    {
        Debug.LogWarning("Capsule radius can not be negative!");
        return;
    }
    // 根据朝向找到up 和 高度缩放值
    Vector3 up = Vector3.up;
    // 半径缩放值
    float radiusScale = 1f;
    // 高度缩放值
    float heightScale = 1f;
    switch (direction)
    {
        case CapsuleDirection.XAxis:
            up = Vector3.right;
            heightScale = Mathf.Abs(scale.x);
            radiusScale = Mathf.Max(Mathf.Abs(scale.y), Mathf.Abs(scale.z));
            break;
        case CapsuleDirection.YAxis:
            up = Vector3.up;
            heightScale = Mathf.Abs(scale.y);
            radiusScale = Mathf.Max(Mathf.Abs(scale.x), Mathf.Abs(scale.z));
            break;
        case CapsuleDirection.ZAxis:
            up = Vector3.forward;
            heightScale = Mathf.Abs(scale.z);
            radiusScale = Mathf.Max(Mathf.Abs(scale.x), Mathf.Abs(scale.y));
            break;
    }

    float realRadius = radiusScale * radius;
    height = height * heightScale;
    float sideHeight = Mathf.Max(height - 2 * realRadius, 0f);

    center = new Vector3(center.x * scale.x, center.y * scale.y, center.z * scale.z);
    // 为了符合Unity的CapsuleCollider的绘制样式,调整位置
    pos = pos - up.normalized * (sideHeight * 0.5f + realRadius) + center;

    Color oldColor = Gizmos.color;
    Gizmos.color = color;

    up = up.normalized * realRadius;
    Vector3 forward = Vector3.Slerp(up, -up, 0.5f);
    Vector3 right = Vector3.Cross(up, forward).normalized * realRadius;

    Vector3 start = pos + up;
    Vector3 end = pos + up.normalized * (sideHeight + realRadius);

    // 半径圆
    DrawCircleImp(start, up, color, realRadius);
    DrawCircleImp(end, up, color, realRadius);

    // 边线
    Gizmos.DrawLine(start - forward, end - forward);
    Gizmos.DrawLine(start + right, end + right);
    Gizmos.DrawLine(start - right, end - right);
    Gizmos.DrawLine(start + forward, end + forward);
    Gizmos.DrawLine(start - forward, end - forward);

    for (int i = 1; i < 26; i++)
    {
        // 下部的头
        Gizmos.DrawLine(start + Vector3.Slerp(right, -up, (i - 1) / 25f), start + Vector3.Slerp(right, -up, i / 25f));
        Gizmos.DrawLine(start + Vector3.Slerp(-right, -up, (i - 1) / 25f), start + Vector3.Slerp(-right, -up, i / 25f));
        Gizmos.DrawLine(start + Vector3.Slerp(forward, -up, (i - 1) / 25f), start + Vector3.Slerp(forward, -up, i / 25f));
        Gizmos.DrawLine(start + Vector3.Slerp(-forward, -up, (i - 1) / 25f), start + Vector3.Slerp(-forward, -up, i / 25f));

        // 上部的头
        Gizmos.DrawLine(end + Vector3.Slerp(forward, up, (i - 1) / 25f), end + Vector3.Slerp(forward, up, i / 25f));
        Gizmos.DrawLine(end + Vector3.Slerp(-forward, up, (i - 1) / 25f), end + Vector3.Slerp(-forward, up, i / 25f));
        Gizmos.DrawLine(end + Vector3.Slerp(right, up, (i - 1) / 25f), end + Vector3.Slerp(right, up, i / 25f));
        Gizmos.DrawLine(end + Vector3.Slerp(-right, up, (i - 1) / 25f), end + Vector3.Slerp(-right, up, i / 25f));
    }

    Gizmos.color = oldColor;
}

用支付宝请我喝咖啡

用微信请我吃辣条

最近的文章

打造舒适的Unity开发环境

前言我相信每个开发人员都在工作中慢慢配置了一套让自己最舒适的开发环境。不管是软件还是硬件,用起来得心应手,剑随意动一直是程序员们的追求。本文想仅从软件的角度介绍下我自己开发基于Unity的游戏时的工具及配置。希望初学者可以根据此文扫清环境搭建的困惑,资深开发者可以与我交流自己的见解。本文将持续维护并优先更新于我的github和博客。系统篇本文采用的是Windows 10。windows系统默认的一些设置不太适合程序员,我们来做一些设置。 (点击Windows键,输入file,)打开文件资...…

继续阅读
更早的文章

Unity预设序列化的自动化方案

预设在Unity中,我们会将重复使用的资源做成预设(Prefab)。预设上可以挂载脚本并在Aspector面板中指定可序列化的值便于配置。在实际操作过程中,我们的脚本上需要配置的部分可能常常需要指定一些固定的组件。加入有这样的一个脚本:像这样,rigBody和capsuleCollider如果在游戏运行时获取会有些消耗,而在Editor模式下又需要我们每次从Hierarchy面板拖动到Aspector面板来指定费时费力。而这一切我们可以通过脚本自动完成。我们在Monster脚本上添加如下接...…

继续阅读