Unity 如何实现框选游戏战斗单位

news/2024/5/4 22:25:58/文章来源:https://blog.csdn.net/qq_42139931/article/details/128058084

文章目录

  • 🍔 Preface
  • ✨ 如何在屏幕坐标系内绘制框选框
  • 🎉 根据框选范围定位其在世界坐标系中对应的区域
  • 🥇 在该区域内进行物理检测


🍔 Preface

本文简单介绍如何实现即时战略游戏中框选战斗单位的功能,如图所示:

框选战斗单位

🍺 实现思路:

本文将该功能的实现拆分为以下部分:

  • 在屏幕坐标系中绘制框选范围;
  • 根据框选范围定位其在世界坐标系中对应的区域;
  • 在该区域内进行物理检测。

✨ 如何在屏幕坐标系内绘制框选框

使用Line Renderer光线渲染器组件来进行范围绘制,当鼠标按下时,可以获得框选范围的起始点,鼠标持续按下时,鼠标位置则是框选范围的结束点,根据这两个点的坐标可以求得另外两个顶点的坐标,如图所示:

框选范围
首先设置Line Renderer光线渲染器的属性:

Line Renderer

  • Enable:默认设为false,当鼠标按下时将其设为true;
  • Loop:设为true,为了让第三个顶点与起始点相连形成闭环;
  • Size:设为4,框选范围有4个顶点;
  • Width:设为0.001即可,线框不需要很粗,可适当调整;

代码部分:

using UnityEngine;
using SK.Framework;
using System.Collections.Generic;public class Example : MonoBehaviour
{//光线渲染器组件private LineRenderer lineRenderer;//屏幕坐标系起始点private Vector3 screenStartPoint;//屏幕坐标系结束点private Vector3 screenEndPoint;private void Start(){//获取光线渲染器组件lineRenderer = GetComponent<LineRenderer>();}private void Update(){//鼠标按下if (Input.GetMouseButtonDown(0)){//激活光线渲染器lineRenderer.enabled = true;//屏幕坐标系起始点screenStartPoint = Input.mousePosition;screenStartPoint.z = 1;}//鼠标持续按下if (Input.GetMouseButton(0)){//屏幕坐标系结束点screenEndPoint = Input.mousePosition;screenEndPoint.z = 1;//求得框选框的另外两个顶点的位置Vector3 point1 = new Vector3(screenEndPoint.x, screenStartPoint.y, 1);Vector3 point2 = new Vector3(screenStartPoint.x, screenEndPoint.y, 1);//接下来使用光线渲染器画出框选范围lineRenderer.SetPosition(0, Camera.main.ScreenToWorldPoint(screenStartPoint));lineRenderer.SetPosition(1, Camera.main.ScreenToWorldPoint(point1));lineRenderer.SetPosition(2, Camera.main.ScreenToWorldPoint(screenEndPoint));lineRenderer.SetPosition(3, Camera.main.ScreenToWorldPoint(point2));}//鼠标抬起if (Input.GetMouseButtonUp(0)){//取消光线渲染器lineRenderer.enabled = false;}}
}

如图所示,已经实现框选范围的绘制:

绘制框选范围

🎉 根据框选范围定位其在世界坐标系中对应的区域

该部分的实现主要依靠物理射线检测,在鼠标位置发出射线,检测与地面的碰撞点,首先为Plane地面设置Layer层级:

地面层级
在鼠标按下时根据射线检测信息确定世界坐标系中的起始点:


//鼠标按下
if (Input.GetMouseButtonDown(0))
{//激活光线渲染器lineRenderer.enabled = true;//屏幕坐标系起始点screenStartPoint = Input.mousePosition;screenStartPoint.z = 1;//射线检测if (Physics.Raycast(mainCamera.ScreenPointToRay(Input.mousePosition), out hit, 1 << LayerMask.NameToLayer("Ground"))){//世界坐标系起始点worldStartPoint = hit.point;}
}

在鼠标抬起时根据射线检测信息确定世界坐标系中的结束点:


//鼠标抬起
if (Input.GetMouseButtonUp(0))
{//取消光线渲染器lineRenderer.enabled = false;//射线检测if (Physics.Raycast(mainCamera.ScreenPointToRay(Input.mousePosition), out hit, 1 << LayerMask.NameToLayer("Ground"))){//世界坐标系结束点worldEndPoint = hit.point;}
}

🥇 在该区域内进行物理检测

该部分用的的核心API:

Physics OverlapBox

可以理解为创建一个碰撞盒来检测该范围内的碰撞体,首先计算出该API需要传入的参数:

  • center:该盒子的中心点;
  • halfExtents:该盒子长宽高的一半。
//盒子中心点
Vector3 center = new Vector3((worldEndPoint.x + worldStartPoint.x) * .5f, 1f, (worldEndPoint.z + worldStartPoint.z) * .5f);
//盒子长宽高的一半
Vector3 halfExtents = new Vector3(Mathf.Abs(worldEndPoint.x - worldStartPoint.x) * .5f, 1f, Mathf.Abs(worldEndPoint.z - worldStartPoint.z) * .5f);

有了这两个参数,调用该API可以获得该区域内的所有碰撞体,遍历判断碰撞体身上如果包含指定的组件,则将其选中,这里使用Outline高亮组件来表示:

//盒子中心点
Vector3 center = new Vector3((worldEndPoint.x + worldStartPoint.x) * .5f, 1f, (worldEndPoint.z + worldStartPoint.z) * .5f);
//盒子长宽高的一半
Vector3 halfExtents = new Vector3(Mathf.Abs(worldEndPoint.x - worldStartPoint.x) * .5f, 1f, Mathf.Abs(worldEndPoint.z - worldStartPoint.z) * .5f);
//检测到盒子内的碰撞器
Collider[] colliders = Physics.OverlapBox(center, halfExtents);for (int i = 0; i < colliders.Length; i++)
{var collider = colliders[i];var outline = collider.GetComponent<Outline>();if (outline != null){outline.enabled = true;}
}

如图所示,我们已经实现了基本的框选功能:

框选功能
在框选时,还需要清除上一次框选的内容,因此我们使用一个List列表来记录当前框选的战斗单位,框选前遍历该列表来清除框选记录,完整代码如下:

public class Example : MonoBehaviour
{//光线渲染器组件private LineRenderer lineRenderer;//屏幕坐标系起始点private Vector3 screenStartPoint;//屏幕坐标系结束点private Vector3 screenEndPoint;//主相机private Camera mainCamera;//碰撞信息private RaycastHit hit;//世界坐标系起始点private Vector3 worldStartPoint;//世界坐标系结束点private Vector3 worldEndPoint;//框选记录列表private List<Outline> list = new List<Outline>();private void Start(){//获取光线渲染器组件lineRenderer = GetComponent<LineRenderer>();//获取主相机mainCamera = Camera.main != null ? Camera.main : FindObjectOfType<Camera>();}private void Update(){//鼠标按下if (Input.GetMouseButtonDown(0)){//激活光线渲染器lineRenderer.enabled = true;//屏幕坐标系起始点screenStartPoint = Input.mousePosition;screenStartPoint.z = 1;//射线检测if (Physics.Raycast(mainCamera.ScreenPointToRay(Input.mousePosition), out hit, 1 << LayerMask.NameToLayer("Ground"))){//世界坐标系起始点worldStartPoint = hit.point;}}//鼠标持续按下if (Input.GetMouseButton(0)){//屏幕坐标系结束点screenEndPoint = Input.mousePosition;screenEndPoint.z = 1;//求得框选框的另外两个顶点的位置Vector3 point1 = new Vector3(screenEndPoint.x, screenStartPoint.y, 1);Vector3 point2 = new Vector3(screenStartPoint.x, screenEndPoint.y, 1);//接下来使用光线渲染器画出框选范围lineRenderer.SetPosition(0, Camera.main.ScreenToWorldPoint(screenStartPoint));lineRenderer.SetPosition(1, Camera.main.ScreenToWorldPoint(point1));lineRenderer.SetPosition(2, Camera.main.ScreenToWorldPoint(screenEndPoint));lineRenderer.SetPosition(3, Camera.main.ScreenToWorldPoint(point2));}//鼠标抬起if (Input.GetMouseButtonUp(0)){//取消光线渲染器lineRenderer.enabled = false;//首先清除上一次的框选记录for (int i = 0; i < list.Count; i++){list[i].enabled = false;}list.Clear();//射线检测if (Physics.Raycast(mainCamera.ScreenPointToRay(Input.mousePosition), out hit, 1 << LayerMask.NameToLayer("Ground"))){//世界坐标系结束点worldEndPoint = hit.point;//盒子中心点Vector3 center = new Vector3((worldEndPoint.x + worldStartPoint.x) * .5f, 1f, (worldEndPoint.z + worldStartPoint.z) * .5f);//盒子长宽高的一半Vector3 halfExtents = new Vector3(Mathf.Abs(worldEndPoint.x - worldStartPoint.x) * .5f, 1f, Mathf.Abs(worldEndPoint.z - worldStartPoint.z) * .5f);//检测到盒子内的碰撞器Collider[] colliders = Physics.OverlapBox(center, halfExtents);for (int i = 0; i < colliders.Length; i++){var collider = colliders[i];var outline = collider.GetComponent<Outline>();if (outline != null){list.Add(outline);outline.enabled = true;}}}}}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.luyixian.cn/news_show_226091.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【外卖项目实战开发二】

文章目录1、完善登录功能问题分析代码实现2、新增员工需求分析数据模型代码开发3、员工信息分页查询需求分析代码开发4、启用/禁用员工账号需求分析代码开发代码修复5、编辑员工信息需求分析代码开发1、完善登录功能 问题分析 前面我们已经完成了后台系统的员工登录功能开发&…

基于JavaWeb的婚恋交友网站设计与实现

项目描述 临近学期结束&#xff0c;还是毕业设计&#xff0c;你还在做java程序网络编程&#xff0c;期末作业&#xff0c;老师的作业要求觉得大了吗?不知道毕业设计该怎么办?网页功能的数量是否太多?没有合适的类型或系统?等等。这里根据疫情当下&#xff0c;你想解决的问…

大数据:HDFS的Shell常用命令操作

文章目录一 HDFS的Shell介绍二 HDFS常用命令操作01 创建目录&#xff08;1&#xff09;创建单层目录&#xff08;3&#xff09;创建多层目录02 查看目录03 上传本地文件到HDFS04 查看文件内容05 下载HDFS文件到本地06 删除HDFS文件07 删除HDFS目录08 移动目录或文件09 文件合并…

第九章 堆排序与TOPK问题

第九章&#xff1a;堆排序与TOPK问题一、堆排序&#xff1a;1、思路分析&#xff1a;&#xff08;1&#xff09;建堆&#xff08;2&#xff09;排序2、堆排序模板二、TOPK问题&#xff1a;1、什么是TOPK问题&#xff1f;2、解决方法一、堆排序&#xff1a; 假设我们实现一个小…

26k Star, 理解Git太轻松了。。。

程序员宝藏库&#xff1a;gitee.com/sharetech_lee/CS-Books-Store Git是目前使用比较广泛一款版本控制工具&#xff0c;从事开发工作&#xff0c;很难绕开Git。 因此&#xff0c;关于如何快速学习Git使用一直都是一个经久不衰的话题。 前不久我在另外一篇文章中曾提到Git对初…

树莓派上搭建SVN服务器

目录 一、服务端安装步骤 1.安装svn 2.创建目录 3.创建版本仓库 4.修改配置&#xff08;authz,passwd,svnserve.conf&#xff09; 5.启动服务 二、tortoisSVN客户端安装 三、结束 一、服务端安装步骤 1.安装svn sudo apt-get install subversion 2.创建目录 sudo m…

用Python蹭别人家图片接口,做一个【免费图床】吧

打开本文&#xff0c;相信你确实需要一个免费且稳定的图床&#xff0c;这篇博客就让你实现。 文章目录⛳️ 谁家的图床⛳️ 实战编码⛳️ 第一轮编码⛳️ 第二轮编码⛳️ 第三轮编码⛳️ 第四轮编码⛳️ 谁家的图床 这次咱们用新浪微博来实现【免费图床应用】&#xff0c;通过…

基于keras 卷积神经外网络搭建的手写数字识别 完整代码+数据可直接运行

项目介绍: 适合新手入门学习代码数据很简洁 上结果: 主要的卷积神经网络: 卷积是指在滑动中提取特征的过程,可以形象地理解为用放大镜把每步都放大并且拍下来,再把拍下来的图片拼接成一个新的大图片的过程。 2D卷积是一个相当简单的操作: 我们先从一个小小的权重矩阵…

十个值得珍藏的正则表达式

正则表达式常学常忘&#xff0c;记规则不如记例子&#xff0c;记多不如记精&#xff0c;记例子就记最经典的。下面是本人珍藏的十个有用的正则表达式&#xff0c;不吝分享&#xff0c;以飨读者。 正则表达式要点 小括号&#xff1a;代表分组 中括号&#xff1a;代表集合 大括号…

外卖项目08---Linux

目录 一、 Linux简介 119 二、Linux安装 120 三、常用命令 122 3.1Linux命令初体验 3.1.1 command [-options] [parameter] 3.2Linux常用命令---文件目录操作命令-ls&-cd&-cat 124 3.2.1list 3.2.2 cd 3.2.3 cat 3.3 Linux常用命令---文件目录操作命令…

机器学习模型与backtrader框架整合

原创文章第116篇&#xff0c;专注“个人成长与财富自由、世界运作的逻辑&#xff0c; AI量化投资”。 北京疫情似乎还没有到拐点&#xff0c;但这三天结束后应该会到来。 今天重点说说&#xff0c;机器学习模型整合到我们的回测框架中&#xff0c;并与backtrader连接起来回测…

傻白入门芯片设计,先进封装技术(五)

集成电路芯片与封装之间是不可分割的整体。没有一个芯片可以不用封装就能正常工作&#xff0c;封装对芯片来说是必不可少的&#xff0c;随着IC生产技术的进步&#xff0c;封装技术也不断更新换代&#xff0c;每一代IC都与新一代的IC封装技术紧密相连。 目录 一、什么是封装&am…

详解设计模式:抽象工厂模式

工厂方法模式&#xff0c;又称工厂模式、多态工厂模式和虚拟构造器模式&#xff0c;通过工厂父类定义负责创建产品的公共接口&#xff0c;子类负责生产具体对象。可以理解为简单工程模式的升级&#xff0c;解决简单工厂模式的弊端。 &#xff5e; 本篇内容包括&#xff1a;关于…

java基本语法 下

目录 运算符 运算符&#xff1a;算术运算符 运算符&#xff1a;赋值运算符 运算符&#xff1a;比较运算符 运算符&#xff1a;逻辑运算符 运算符&#xff1a;三元运算符 运算符的优先级 程序流程控制 概念 顺序结构 if-else结构 switch-case结构 循环结构 循环结构…

你不能错过的【Python爬虫】测试3(爬取所有内容 + 完整源代码 + 架构 + 结果)

目录 一、主要工具包 以及 版本二、架构展示三、各部分code3.1 yjs.py (重要)3.2 items.py3.3 middlewares.py3.4 pipelines.py3.5 settings.py3.6 start.py四、结果展示一、主要工具包 以及 版本 scrapy:2.7.1版本(这里主要用到的工具包) 二、架构展示 三、各部分code 3…

8、MyBatis核心配置文件之typeAliases(mybatis-config.xml)

MyBatis核心配置文件之typeAliases&#xff08;mybatis-config.xml&#xff09; 1、&#xff01;&#xff01;&#xff01;&#xff01;注意 2、 设置类型别名&#xff08;比如有的全类名&#xff08;resultType&#xff09;太长了不好使用&#xff09; typeAlias :设置某个类…

AOP实现方式-P20,21,22

项目的包&#xff1a; pom依赖导入有关aop的包&#xff1a; <dependencies><!-- https://mvnrepository.com/artifact/org.aspectj/aspectjweaver --><dependency><groupId>org.aspectj</groupId><artifactId>aspectjweaver</artifactI…

mysql-6-主从复制搭建

1 总结 1&#xff1a;主从复制最大缺陷就是延迟。 2 搭建前的准备 2.1复制的基本原则 每个slave只有一个master每个slave只能有一个唯一的服务器ID每个master可以有多个slavemysql版本尽量一致&#xff0c;防止出问题。两台服务能ping通MySQL主从是基于binlog的&#xff0c;主上…

【路径规划】(1) Dijkstra 算法求解最短路,附python完整代码

好久不见&#xff0c;我又回来了&#xff0c;这段时间把路径规划的一系列算法整理一下&#xff0c;感兴趣的点个关注。今天介绍一下机器人路径规划算法中最基础的 Dijkstra 算法&#xff0c;文末有 python 完整代码&#xff0c;那我们开始吧。 1. 算法介绍 1959 年&#xff0c…

FAIRNESS IN MACHINE LEARNING: A SURVEY 阅读笔记

论文链接 刚读完一篇关于机器学习领域研究公平性的综述&#xff0c;这篇综述想必与其有许多共通之处&#xff0c;重合部分不再整理笔记&#xff0c;可详见上一篇论文的笔记&#xff1a; A Survey on Bias and Fairness in Machine Learning 阅读笔记_Catherine_he_ye的博客 S…