Unity实现攻击范围检测并绘制检测区域
作者:画个小圆儿 发布时间:2023-09-18 21:59:10
标签:unity,攻击,检测
本文实例为大家分享了Unity实现攻击范围检测并绘制检测区域的具体代码,供大家参考,具体内容如下
一、圆形检测
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
/// <summary>
/// 圆形检测,并绘制出运行的攻击范围
/// </summary>
public class CircleDetect : MonoBehaviour {
GameObject go; //生成矩形的对象
public Transform attack; //被攻击方
MeshFilter mf;
MeshRenderer mr;
Shader shader;
void Start () {
}
void Update () {
if (Input.GetKeyDown(KeyCode.A))
{
ToDrawCircleSolid(transform, transform.localPosition, 3);
if (CircleAttack(attack,transform,3))
{
Debug.Log("攻击到了");
}
}
if (Input.GetKeyUp(KeyCode.A))
{
if (go != null)
{
Destroy(go);
}
}
}
/// <summary>
/// 圆形检测
/// </summary>
/// <param name="attacked">被攻击者</param>
/// <param name="skillPostion">技能的位置</param>
/// <param name="radius">半径</param>
/// <returns></returns>
public bool CircleAttack(Transform attacked, Transform skillPostion, float radius)
{
float distance = Vector3.Distance(attacked.position, skillPostion.position);
if (distance <= radius)
{
return true;
}
else
{
return false;
}
}
//生成网格
public GameObject CreateMesh(List<Vector3> vertices)
{
int[] triangles;
Mesh mesh = new Mesh();
int triangleAmount = vertices.Count - 2;
triangles = new int[3 * triangleAmount];
//根据三角形的个数,来计算绘制三角形的顶点顺序
//顺序必须为顺时针或者逆时针
for (int i = 0; i < triangleAmount; i++)
{
triangles[3 * i] = 0;
triangles[3 * i + 1] = i + 1;
triangles[3 * i + 2] = i + 2;
}
if (go == null)
{
go = new GameObject("circle");
go.transform.SetParent(transform, false);
go.transform.position = new Vector3(0, -0.4f, 0);
mf = go.AddComponent<MeshFilter>();
mr = go.AddComponent<MeshRenderer>();
shader = Shader.Find("Unlit/Color");
}
//分配一个新的顶点位置数组
mesh.vertices = vertices.ToArray();
//包含网格中所有三角形的数组
mesh.triangles = triangles;
mf.mesh = mesh;
mr.material.shader = shader;
mr.material.color = Color.red;
return go;
}
/// <summary>
/// 绘制实心圆形
/// </summary>
/// <param name="t">圆形参考物</param>
/// <param name="center">圆心</param>
/// <param name="radius">半径</param>
public void ToDrawCircleSolid(Transform t, Vector3 center, float radius)
{
int pointAmount = 100;
float eachAngle = 360f / pointAmount;
Vector3 forward = t.forward;
List<Vector3> vertices = new List<Vector3>();
for (int i = 0; i < pointAmount; i++)
{
Vector3 pos = Quaternion.Euler(0f, eachAngle * i, 0f) * forward * radius + center;
vertices.Add(pos);
}
CreateMesh(vertices);
}
}
效果图:
二、矩形检测
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
/// <summary>
/// 矩形型攻击检测,并绘制检测区域
/// </summary>
public class DrawRectangDetect : MonoBehaviour {
public Transform attacked;
GameObject go; //生成矩形
MeshFilter mf;
MeshRenderer mr;
Shader shader;
void Start () {
}
void Update () {
if (Input.GetKeyDown(KeyCode.A))
{
ToDrawRectangleSolid(transform, transform.localPosition, 4, 2);
if (RectAttackJubge(transform, attacked, 4, 2f))
{
Debug.Log("攻击到");
}
}
if (Input.GetKeyUp(KeyCode.A))
{
if (go != null)
{
Destroy(go);
}
}
}
/// <summary>
/// 矩形攻击范围
/// </summary>
/// <param name="attacker">攻击方</param>
/// <param name="attacked">被攻击方</param>
/// <param name="forwardDistance">矩形前方的距离</param>
/// <param name="rightDistance">矩形宽度/2</param>
/// <returns></returns>
public bool RectAttackJubge(Transform attacker, Transform attacked, float forwardDistance, float rightDistance)
{
Vector3 deltaA = attacked.position - attacker.position;
float forwardDotA = Vector3.Dot(attacker.forward, deltaA);
if (forwardDotA > 0 && forwardDotA <= forwardDistance)
{
if (Mathf.Abs(Vector3.Dot(attacker.right,deltaA)) < rightDistance)
{
return true;
}
}
return false;
}
//制作网格
private GameObject CreateMesh(List<Vector3> vertices)
{
int[] triangles;
Mesh mesh = new Mesh();
int triangleAmount = vertices.Count - 2;
triangles = new int[3 * triangleAmount];
for (int i = 0; i < triangleAmount; i++)
{
triangles[3 * 1] = 0;
triangles[3 * i + 1] = i + 1;
triangles[3 * i + 2] = i + 2;
}
if (go == null)
{
go = new GameObject("Rectang");
go.transform.position = new Vector3(0, 0.1f, 0);
mf = go.AddComponent<MeshFilter>();
mr = go.AddComponent<MeshRenderer>();
shader = Shader.Find("Unlit/Color");
}
mesh.vertices = vertices.ToArray();
mesh.triangles = triangles;
mf.mesh = mesh;
mr.material.shader = shader;
mr.material.color = Color.red;
return go;
}
/// <summary>
/// 绘制实心长方形
/// </summary>
/// <param name="t">矩形参考物</param>
/// <param name="bottomMiddle">矩形的中心点</param>
/// <param name="length">矩形长度</param>
/// <param name="width">矩形宽度的一半</param>
public void ToDrawRectangleSolid(Transform t, Vector3 bottomMiddle, float length, float width)
{
List<Vector3> vertices = new List<Vector3>();
vertices.Add(bottomMiddle - t.right * width);
vertices.Add(bottomMiddle - t.right * width + t.forward * length);
vertices.Add(bottomMiddle + t.right * width + t.forward * length);
vertices.Add(bottomMiddle + t.right * width );
CreateMesh(vertices);
}
}
效果图:
三、扇形攻击检测
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
/// <summary>
/// 扇型攻击检测,并绘制检测区域
/// </summary>
public class SectorDetect : MonoBehaviour {
public Transform attacked; //受攻击着
GameObject go;
MeshFilter mf;
MeshRenderer mr;
Shader shader;
void Start () {
}
void Update () {
if (Input.GetKeyDown(KeyCode.A))
{
ToDrawSectorSolid(transform, transform.localPosition, 60, 3);
if (UmbrellaAttact(transform,attacked.transform,60,4))
{
Debug.Log("受攻击了");
}
}
if (Input.GetKeyUp(KeyCode.A))
{
if (go != null)
{
Destroy(go);
}
}
}
/// <summary>
/// 扇形攻击范围
/// </summary>
/// <param name="attacker">攻击者</param>
/// <param name="attacked">被攻击方</param>
/// <param name="angle">扇形角度</param>
/// <param name="radius">扇形半径</param>
/// <returns></returns>
public bool UmbrellaAttact(Transform attacker, Transform attacked, float angle, float radius)
{
Vector3 deltaA = attacked.position - attacker.position;
//Mathf.Rad2Deg : 弧度值到度转换常度
//Mathf.Acos(f) : 返回参数f的反余弦值
float tmpAngle = Mathf.Acos(Vector3.Dot(deltaA.normalized, attacker.forward)) * Mathf.Rad2Deg;
if (tmpAngle < angle * 0.5f && deltaA.magnitude < radius)
{
return true;
}
return false;
}
public void ToDrawSectorSolid(Transform t, Vector3 center, float angle, float radius)
{
int pointAmmount = 100;
float eachAngle = angle / pointAmmount;
Vector3 forward = t.forward;
List<Vector3> vertices = new List<Vector3>();
vertices.Add(center);
for (int i = 0; i < pointAmmount; i++)
{
Vector3 pos = Quaternion.Euler(0f, -angle / 2 + eachAngle * (i - 1), 0f) * forward * radius + center;
vertices.Add(pos);
}
CreateMesh(vertices);
}
private GameObject CreateMesh(List<Vector3> vertices)
{
int[] triangles;
Mesh mesh = new Mesh();
int triangleAmount = vertices.Count - 2;
triangles = new int[3 * triangleAmount];
//根据三角形的个数,来计算绘制三角形的顶点顺序
for (int i = 0; i < triangleAmount; i++)
{
triangles[3 * i] = 0;
triangles[3 * i + 1] = i + 1;
triangles[3 * i + 2] = i + 2;
}
if (go == null)
{
go = new GameObject("mesh");
go.transform.position = new Vector3(0f, 0.1f, 0.5f);
mf = go.AddComponent<MeshFilter>();
mr = go.AddComponent<MeshRenderer>();
shader = Shader.Find("Unlit/Color");
}
mesh.vertices = vertices.ToArray();
mesh.triangles = triangles;
mf.mesh = mesh;
mr.material.shader = shader;
mr.material.color = Color.red;
return go;
}
}
效果图:
来源:https://blog.csdn.net/qq_38721111/article/details/86742317


猜你喜欢
- 这篇文章主要介绍了SpringBoot集成MybatisPlus报错的解决方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定
- 本文实例为大家分享了Android实现屏幕录制功能的具体代码,供大家参考,具体内容如下1.效果图:2.添加依赖 dependenc
- 本文实例为大家分享了java图形用户界面实现菜单功能的具体代码,供大家参考,具体内容如下题目:编写一个图形用户界面,实现菜单的功能。有3个一
- 1.首先,八种基本数据类型分别是:int、short、float、double、long、boolean、byte、char; &
- 目前的应用市场上,使用毛玻璃效果的APP随处可见,比如用过微信语音聊天的人可以发现,语音聊天页面就使用了高斯模糊效果。先看下效果图:&nbs
- 前言兄弟们,刚刚又给seata社区修了一个BUG,有用户提了issue反应TransactionHook在某些情况下不会被调用:相关issu
- private void button1_Click(object sender, EventArgs e) {
- Seata 是一款开源的分布式事务解决方案,致力于提供高性能和简单易用的分布式事务服务。本篇不涉及其原理,只用代码构建项目简单试用一下其回滚
- 一.前言RabbitMQ的TTL全称为Time-To-Live,表示的是消息的有效期。消息如果在队列中一直没有被消费并且存在时间超过了TTL
- 本文实例讲述了C#线性渐变画刷LinearGradientBrush用法。分享给大家供大家参考。具体如下:using System;usin
- 一般的接口实现多态定义接口 interface Ipeople { void say(); }定义实现的类 public cla
- MyBatis 获取子类的属性这里有个model类:基类public class user { pu
- 感想:第一次写博客,感觉编辑器挺复杂厉害的,感觉自己的内容挺简单的。有什么问题请多多指教!思路:1、创建一个扑克牌的实体类Poker,设置了
- 1 SeekBar简介SeekBar是进度条。我们使用进度条时,可以使用系统默认的进度条;也可以自定义进度条的图片和滑块图片等。2 Seek
- 在手机客户端尤其是Android应用的开发过程中,我们经常会接触到“硬件加速”这个词。由于操作系统对底层软硬件封装非常完善,上层软件开发者往
- 本文为大家分享了Java多线程实现Runnable方式的具体方法,供大家参考,具体内容如下(一)步骤 1.定义实现Runnable
- Spring Cache设置缓存条件原理从Spring3.1开始,Spring框架提供了对Cache的支持,提供了一个对缓存使用的抽象,通过
- 目录对象的创建创建方式对象的内存布局创建过程对象的内存分配分配方式并发安全代码优化逃逸分析的不成熟性实际的对象空间分配过程对象的访问句柄直接
- 代理对象的生成方法是:Proxy.newProxyInstance(...) ,进入这个方法内部,一步一步往下走会发现会调用ProxyGen
- 前面我们讲到了Spring在进行事务逻辑织入的时候,无论是事务开始,提交或者回滚,都会触发相应的事务事件。本文首先会使用实例进行讲解Spri