概述
交互功能使用同一个按钮或按钮列表,在不同情况下显示不同的内容,按下执行不同的操作。
按选项个数分类
环境交互系统可分为两种,单选项交互,一般使用射线检测;多选项交互,一般使用范围检测。第一人称游戏单选多选都可以用,因为第一人称人物背对一个可交互对象时显示交互选项让玩家疑惑,所以第一人称使用射线检测或使用人物面前的触发器进行范围检测。第三人称因为人物一般在或者接近画面中心,基本上必须用以人物为中心的范围检测。
本质上,单选项交互让玩家通过转动视角选择交互选项,在可交互对象很密集的情景,玩家很难准确瞄准交互对象,但开发简单。多选项交互玩家选择想要的交互选项更容易,开发复杂。
按输入方式分类
电脑通过键盘按键(一般是F)交互,滑轮选选项;手机通过按屏幕交互,上下滑动选选项,这导致二者的实现方式又不一样。电脑通过InputManager或PlayerInput,手机通过按钮回调。就是说电脑上的多个选项可以只是Image,手机上的选项有Button。
环境交互系统由这几个部分构成:
- 环境检测;
- 交互类型判断;
- UI显示和回调添加;
- 执行;
对于玩家,他能直接感觉到的只有UI显示和按下执行,环境检测、根据类型判断UI显示的内容、根据类型判断执行的类型完全由程序完成,所以玩家感觉不到交互系统的复杂性。
数据结构
一个游戏的交互类型选项是确定的,有限的,适合用枚举表示。把游戏支持的所有交互类型定义为枚举的选项:
public enum ActionOption{None,TakeGun,SwapGun,TakeItem,Talk,CheckPack
}
如果是单选项交互,就声明一个交互枚举变量,如果是多选项交互,就声明一个交互枚举列表。
数据结构的维护
对于单选项交互,程序每一帧都执行环境检测,更新交互信息变量,开销还可以接受。
多选项交互如果每帧都删掉交互列表再重新获得开销太大,可以在OnTriggerEnter()添加,OnTriggerExit()移除,但因为执行交互后需要对交互列表更新,此时的更新与触发器进入退出无关,所以清空更新交互列表的函数肯定要写。而根据测试,执行拿起物品同时放下身上物品会触发OnTriggerEnter(),如果在OnTriggerEnter()时添加交互项,执行交互后又更新交互项,那么执行拿起并放下物品时拿起刚被放下的物品的交互项就会多一个。
综合起来:清空交互列表再更新比较稳健,不会造成列表多余或缺少交互项(这个结论适用于任何列表的维护),但每帧更新交互列表开销又大,所以可以在OnTriggerEnter()、OnTriggerExit()和执行交互后三个时机更新交互项列表。
然后又发现在执行一个物体A的OnTriggerExit()里面执行Physics.OverlapSphere()范围和触发器范围一样,还能检测到A,导致离开物体的交互区域时捡起它的交互选项没有消失。不过很快又发现原来是Physics.OverlapSphere()的中心点和触发器的中心点不一致。
至此交互项列表的维护问题才基本解决。就是解决两个问题:1.防止交互项列表比实际的多余或缺失;2.防止每帧清空刷新造成的巨大开销。
环境检测
环境交互函数通过射线检测或范围检测得到碰撞体,从中筛选出可交互对象,使用分类函数得到这个交互的类型、对象名称,以此得到显示在界面上的文本、按下需要执行的操作。
筛选可交换对象可以反复用TryGetComponent()把可能有交互的脚本都判断一遍,但因为可交互对象可能是一个物品、一个NPC、一扇门,交互对象类型虽然毫不相干,但是都有“可交互”的特征。这很符合接口的设计初衷,所以可以定义接口:
public interface Interactive
{public abstract void InterAct();
}
所有的可交互对象继承可交互接口。射线或范围检测只要在检测到的碰撞体上尝试获得接口,就能得到所有可交互对象。
if(Physics.Raycast(rayOrigin,rayVector,out raycastHit,10,interactionLayerMask)&&Vector3.Distance(player.transform.position,raycastHit.point)<interactionRange){if(raycastHit.collider.TryGetComponent(out interactive)){}
}
再把这个接口变量交给一个函数使用连续的if(interactive is xxx)判断它的具体类型 。
对话交互项的检测
如果玩家站在NPC背后允许发起对话,效果会很奇怪。所以NPC的继承interactive的脚本不应该放在人物身上,而在人物面前的节点,上面挂载一个触发器,最好是扇形的,但Unity没有扇形,只好用盒型或球形代替。
UI显示
界面上放一个显示交互选项列表的锚点,挂载Vertical Layout Group组件,制作一个交互选项预制体,由交互管理器脚本把预制体实例化到锚点下。
用滚轮改变选中的交互选项:使用了InputSystem系统。
void Update(){GetScrollInput();}public void GetScrollInput(){float delta=Mouse.current.scroll.ReadValue().y;if(delta<0){selectedInteraction++;}else if(delta>0){selectedInteraction--;}selectedInteraction=Mathf.Clamp(selectedInteraction,0,actionOptions.Count-1);UpdateSelectedInteraction();}
更新显示交互选项:使用白色突出选中项,未选中项使用灰色。
void UpdateSelectedInteraction(){for(int i=0;i<interactionInfos.Count;i++){Text text=interactionInfos[i].actionOptionGO.GetComponentInChildren<Text>();if(i==selectedInteraction){text.color=Color.white;}else{text.color=Color.gray;}}}
执行
使用switch()判断交互的类型执行相应的操作。
和对话系统的关系
对于对话时锁定视角,显示鼠标的对话系统,会需要对话时使交互功能失效。这包括1.隐藏交互菜单;2.关闭对F执行交互的响应;3.最好能关闭对滑轮改变选中的交互的响应。可以给交互管理器写一个开启关闭函数,关闭相应F需要声明一个bool,执行时判断一下再执行。