[Unity3D]自己动手重制坦克舰队ArmadaTank(2)从碰撞说起
[Unity3D]自己动手重制坦克舰队ArmadaTank(2)从碰撞说起
在上一篇里我给出了重制的坦克舰队效果图和试玩程序。本篇介绍一下玩家坦克和敌方坦克碰撞问题。
我们需要什么样的碰撞
原版里,玩家与其它坦克碰撞时,玩家与对方都不能移动;而敌方坦克之间相互碰撞时,是无视碰撞直接穿透的。这些功能的实现需要一些特殊的设计。
需要注意到,坦克舰队里的坦克移动方式是以格为单位的,每次移动都会移动完整的1个单位。就是说,坦克在下图所示的D字母里面只有上边和下边两个停留位置,而不会在中间某处停留。

据此,我们给出实现方法。
障碍物检测法
在游戏画面中,看到的坦克是这样的:

但是为了实现原版的碰撞效果,我们给每个坦克都加上前后左右4个cube作为障碍物检测器(Obstacle Detector),让他们分别检测坦克所在位置的前后左右是否存在障碍物。

为了避免与子弹碰撞,我们把这些cube压扁了,其高度值只有0.1,而长宽仍为1。
此外,这些cube最终目的是检测坦克的存在,所以要给坦克本身一个Collider。

注意,这里我们让敌方坦克自身的Collider半径稍微小于0.5,以便后续的实现。

而玩家坦克的Collider半径保持0.5不变。
你会看到,这样的微小差异,会使敌方坦克之间的碰撞检测稍微迟钝一点,所以他们就可以相互穿透;而敌方坦克与玩家坦克之间仍然不可穿透。
Cube的作用
cube的作用是这样的:每个cube内部维护一个List<GameObject> obstacles列表,这个列表记录此检测器遇到的所有障碍物。每当触发OnTriggerEnter(Collider other)时,就把other指定的物体(障碍物)加入obstacles列表;每当触发OnTriggerExit(Collider other)时,就把other指定的物体(障碍物)移出obstacles列表。这样,就可以通过分析obstacles里的情况判断出此cube代表的方向(前后左右之一)是否有障碍物了。
public abstract class ObstacleDetector : MonoBehaviour {
public System.Collections.Generic.List<GameObject> obstacles;
void Awake()
{
if (obstacles == null)
{
obstacles = new System.Collections.Generic.List<GameObject>();
}
}
protected abstract void OnTriggerEnter(Collider other);
void OnTriggerExit(Collider other)
{
this.obstacles.Remove(other.gameObject);
}
public bool IsUnblocked()
{
for (int i = ; i < this.obstacles.Count; i++)
{
if(this.obstacles[i] != null && this.obstacles[i].collider.enabled)
{
return false;
}
}
return true;
}
}
敌方坦克的障碍物检测器
所有类型的敌方坦克的碰撞效果都相同,所以统一用下面的脚本:
public class EnemyObstacleDetector : ObstacleDetector {
protected override void OnTriggerEnter(Collider other)
{
var tag = other.tag;
if (tag != null)
{
if (tag == Tags.EnemyObstacleDetector) { return; }
if (tag == Tags.PlayerObstacleDetector) { return; }
// 对敌方坦克来说,另一个敌方坦克是可以穿透过去的,所以不应加入obstacles列表。
if (tag == Tags.EnemyObstacleDetectorCenter) { return; }
}
this.obstacles.Add(other.gameObject);
}
}
对敌方坦克来说,另一个敌方坦克是可以穿透过去的,所以不应加入obstacles列表。
如果碰到的是别人的cube(障碍物检测器),直接忽略即可。
玩家坦克的障碍物检测器
玩家坦克与敌方坦克的碰撞效果不同,所以需要单独处理。
public class PlayerObstacleDetector : ObstacleDetector
{ protected override void OnTriggerEnter(Collider other)
{
var tag = other.tag;
if (tag != null)
{
if (tag == Tags.EnemyObstacleDetector) { return; }
if (tag == Tags.PlayerObstacleDetector) { return; }
} this.obstacles.Add(other.gameObject);
} }
对玩家坦克来说,其它坦克是不能穿透的。
同样,如果碰到的是别人的cube(障碍物检测器),直接忽略即可。
继承与GetComponent
ObstacleDetector是一个抽象基类,敌方坦克和玩家坦克分别使用的是其子类EnemyObstacleDetector和PlayerObstacleDetector。此时,GetComponent<T>()仍然可以正常使用。它返回一个基类对象的引用,此引用实际指向的则是某个子类的对象。
这是一个通用的知识。
public class TankTranslate : MonoBehaviour
{
private System.Collections.Generic.Dictionary<TankToward, ObstacleDetector> obstacleDetectorDict; // Use this for initialization
void Start()
{
if (obstacleDetectorDict == null)
{ obstacleDetectorDict = new System.Collections.Generic.Dictionary<TankToward, ObstacleDetector>(); }
var names = new string[] { "forward", "backward", "left", "right" };
var direction = new TankToward[] { TankToward.Z, TankToward.NZ, TankToward.NX, TankToward.X };
for (int i = ; i < names.Length; i++)
{
var child = this.transform.FindChild(names[i]);
var script = child.GetComponent<ObstacleDetector>();
obstacleDetectorDict.Add(direction[i], script);
}
}
}
起火冒烟
坦克被击中会损失health值,视觉效果上我们用冒烟的浓度显示。
冒烟效果用particle System组件来做。

控制冒烟的浓度需要一个脚本。
public class TankSmoke : MonoBehaviour
{
public TankHealth tankHealthScript;
private float lastHealth; // Update is called once per frame
void Update()
{
if (lastHealth != tankHealthScript.health)
{
var lostHealth = (TankHealth.maxHealth - tankHealthScript.health);
if (lostHealth <= )
{
this.particleSystem.enableEmission = false;
}
else
{
this.particleSystem.Play();
this.particleSystem.enableEmission = true;
this.particleSystem.emissionRate = * lostHealth / TankHealth.maxHealth;
}
lastHealth = tankHealthScript.health;
}
}
}
原版坦克舰队里的冒烟素材图片是这样的:

为了正常显示,需要设置其Shader为Mobile Particles/Alpha Blended,并设置Tiling和Offset如下图所示。

如果您不了解Tiling和Offset的含义,可以参考这里(图文详解Unity3D中Material的Tiling和Offset是怎么回事)。
如果您需要项目源码请通过下方二维码捐赠10元并留下您的联系方式。
ps:感谢上次捐赠的人,虽然只有1人。此文作为回馈,但愿您满意。
pps:我更换了博客皮肤,自我感觉比较良好。
[Unity3D]自己动手重制坦克舰队ArmadaTank(2)从碰撞说起的更多相关文章
- [Unity3D]自己动手重制坦克舰队ArmadaTank
[Unity3D]自己动手重制坦克舰队ArmadaTank 我玩过一款坦克游戏ArmadaTank(坦克舰队),如下图所示 几个月前我尝试用Unity3D重制这款游戏,已经可以玩起来了.下面是在PC上 ...
- Xamarin.Forms之UserDialogs 重制版本
在 forms 里面,目前使用比较多的弹出组件是 Acr.UserDialogs ,但是这个组件有些小问题,比如 loading .hide 会同时把 toast 给一起关掉,android 下的 t ...
- 使用AndEngine重制《是男人就上一百层》
1.为什么还要做<是男人就上一百层> 一是在用Android原生API开发完<是男人就上一百层>以后,一直想体验一下用引擎开发游戏是个什么感觉,顺便也让游戏听上去高大上一些(使 ...
- 《杜增强讲Unity之Tanks坦克大战》8-子弹碰撞处理
8 子弹碰撞处理 为了处理子弹打到坦克的伤害我们在这里新建一个Shell.cs 子弹有两种情况,碰到坦克炸开,没有碰到坦克则过2s子弹销毁. void Start () { Destroy (game ...
- 重制AdvanceWars第一步 -- 搞定地图
首先来聊下高级战争吧Advance Wars,由任天堂旗下的Intelligent Systems开发的战棋游戏.初作诞生于GBA上,后来继续跟进了高战2黑洞崛,而后在下一代掌机DS上也出了三代续作高 ...
- 替换/重制Homebrew源
homebrew主要分两部分:git repo(位于GitHub)和二进制bottles(位于bintray),这两者在国内访问都不太顺畅.可以替换成国内的镜像,git repo国内镜像就比较多了,可 ...
- 微软重制Windows 1.0系统:祖师爷出山了
Windows官方推特在7月1日发布了一条很有趣的动态,“向大家介绍全新的Windows 1.0,带MS-DOS.时钟等”.配发的视频回顾了从Windows 1.0/3.1到Windows 10期间, ...
- BIT祝威博客汇总(Blog Index)
+BIT祝威+悄悄在此留下版了个权的信息说: 关于硬件(Hardware) <穿越计算机的迷雾>笔记 继电器是如何成为CPU的(1) 继电器是如何成为CPU的(2) 关于操作系统(Oper ...
- Unity3D 回合制 网上源码 目前还在研究构思
我们已将回合制的战斗模式讲解得很清楚了.那么,如果在Unity3D游戏中实现一个回合制游戏呢?我们从最简单的一对一模式来设计回合制游戏的原型.我们可以游戏的状态划分为下面三种状态: 1. ...
随机推荐
- Andriod学习笔记2:“Your content must have a ListView whose id attribute is 'android.R.id.list'”问题的解决办法
问题描述 activity_main.xml代码如下: <?xml version="1.0" encoding="utf-8"?> <Lin ...
- View动画和属性动画
在应用中, 动画效果提升用户体验, 主要分为View动画和属性动画. View动画变换场景图片效果, 效果包括平移(translate), 缩放(scale), 旋转(rotate), 透明(alph ...
- z-stack组网过程
z-stack组网分:协调器建立网络.路由器和终端加入网络 暂时只记录第一次上电建立网络的过程,至于开启NV_RESTORE后,恢复原有的网络则暂时不分析. 一.协调器建立网络: 1.ZDO层的ZDA ...
- python 小程序 比较目录间的差异
比较目录间的差异: I 只按照名称做了比较,如果目录的文件名称相同,但是内容不同脚本认为为相同文件 II 针对目录下面的目录没有循环比较,只是比较了目录的名称 import sys, os def d ...
- 四则运算app代码上传GITHUB
团队: 郭志豪:http://www.cnblogs.com/gzh13692021053/ 杨子健:http://www.cnblogs.com/yzj666/ 刘森松:http://www.cnb ...
- java function retry wrapper
import java.util.concurrent.Callable; /** * Created by huahui.yang on 1/29/16. */ public class Retry ...
- 巴特沃斯(Butterworth)滤波器 (2) - 双线性变换
这里接着上篇讲一下双线性变换Bilinear Transformation,它实现了模拟信号(连续域)与数字信号(离散域)之间的转换. 双线性变换公式如下: 反推可得到: 因此可以根据连续域传递函数推 ...
- Spring AOP详解
一.前言 在以前的项目中,很少去关注spring aop的具体实现与理论,只是简单了解了一下什么是aop具体怎么用,看到了一篇博文写得还不错,就转载来学习一下,博文地址:http://www.cnbl ...
- TopCoder SRM 639 Div.2 500 AliceGameEasy
题意: 一个游戏有n轮,有A和B比赛,谁在第 i 轮得胜,就获得 i 分,给出x,y,问A得x分,B得y分有没有可能,如果有,输出A最少赢的盘数 解题思路: 首先判断n(n+1)/2 = (x+y)是 ...
- Lession1 写在机器学习之前
机器学习从学习方式上来讲,可以分为两类: 监督学习(Supervised Learning),简而言之就是“有标签”学习 无监督学习(Unsupervised Learning),简而言之就是“无标签 ...