~ [目录] ~
- 0. 前言
- 1. 红点系统
- (1)种类
- (2)结构
- (3)约定
- 2. 红点树
- (1)树节点
- (2)树_增删查
- (3)树_红点处理
- 3. 封装、检查
- (1)检查
- (2)UGF封装为组件
- 4. 结束咯
0. 前言
刚好处理到红点系统的问题,就写个文章记录一下。本文的红点系统为一个树结构,UI实现需要和红点运行逻辑剥离,防止过度耦合,现在就暂时不提及,后续在讲述。
1. 红点系统
红点是游戏中一种常见且重要的提醒方式,通常涉及到很多方面信息和界面的显示,如果不做成独立的系统的话,那么游戏逻辑将会杂乱散落在各个角落不便于开发维护。
(1)种类
红点通常是指有UI上那种带数字的小红标,或者单纯一个小红点。当然也不止这些,比如金矿建筑有无产出的信息,也可以用红点系统来处理,有的话显示个小金币图标在建筑头上。当然这个产出信息通常是和金矿建筑高度关联的,也会让金矿建筑自己来处理。这个可能要先考虑一下?
(2)结构
对于红点的信息来说,可能如下图:
主菜单--背包--道具页--新道具1
--新道具2
--任务--新任务1
--新任务2
通常会呈现一个树状,的结构。
这个时候,子节点有一个点亮,父节应点亮。比如有”新道具1”,”主菜单”、”背包”、”道具页面”、”新道具1”应该都点亮,去引导玩家往去查看这个新道具。查看完之后,”新道具1”子节熄灭,其他父节点也应该重新检查是否点亮。
需要考虑的另一点是,叶节点可以自由控制点亮,父节点是否点亮由子节点决定,所有子节点熄灭,父节点应熄灭。(叶节点:树的末端节点,“新道具1”、“新道具2”、“新任务1”、"新任务2"为叶节点)。”道具页”是否点亮,是由有没有新道具来决定的。如果没有新道具,道具页自己亮了,只会给玩家提供错误的信息。若有需求是只让玩家看看这个道具页,比如道具页面更新了,那么应该另外设置一个子节点来处理。
(3)约定
这里我们约定一下红点由路径节点的值组成,并由“|”分隔,方便后续处理。
比如主菜单–任务–新任务1,这个红点查询时,应查询"menu|Task|newTask1"
2. 红点树
我们可以用树的数据结构来实现这个红点系统,那先来实现一个树结构
(1)树节点
所以是树节点的基础结构,值 + 子节点;
public class RedHitNode
{
public int value;
public List<RedHitNode> children = new List<RedHitNode>();
}
这里采用了List而不是Dictionary主要是考虑到Dictionary开销更大,而按照树结构,子节点通常不会太多,所以用List即可。另外作为红点的节点,我还希望:
- 节点自己控制节点的增加和删除
- 节点是否为叶子节点
- 节点通过一个只读的key来互相区分(在同节点下)
所以给RedHitNode加上Key,以及Add,Remove,Find的功能,那么就得到
public class RedHitNode
{
public readonly string Key;
private List<RedHitNode> children = new List<RedHitNode>();
public int Value = 0;
public bool IsLeaf => children.Count == 0;
public RedHitNode(string key)
{
this.Key = key;
}
public RedHitNode Add(string key)
{
RedHitNode node = null;
int index = IndexOf(key);
if (index == -1)
{
node = new RedHitNode(key);
children.Add(node);
}
return node;
}
public RedHitNode Remove(string key)
{
RedHitNode node = null;
int index = IndexOf(key);
if (index != -1)
{
// 删除对应节点
// 因为无序,所以做了位置互换,防止数据过多复制
children[index] = children[children.Count - 1];
children.RemoveAt(children.Count - 1);
}
return node;
}
public RedHitNode Find(string key)
{
RedHitNode node = null;
int index = IndexOf(key);
if (index != -1)
{
node = children[index];
}
return node;
}
protected int IndexOf(string key)
{
for (int i = 0; i < children.Count; i++)
{
if (children[i].Key == key)
{
return i;
}
}
return -1;
}
}
Remove作为和其他List移除略有不同的是,先将数据移动到末端再删除。因为作为红点数据,其子节点是无序的,所以可以通过这种方式,防止数据删除后,后面的数据往前复制。
(2)树_增删查
作为树,应该具备基本的根节点Root,以及Insert,Delete,Search处理。而且因为作为红点树,其节点值应该完全由自己控制,所以节点的插入删除不对外暴露。
public class RedHitNode
{
private RedHitNode root;
private RedHitNode Insert(string[] keys){}
private RedHitNode Delete(string[] keys){}
private RedHitNode Search(string[] keys){}
}
首先Search就不断查找子节点直到找到,Delete也差不多,找到了就删除。实现如下
private RedHitNode Delete(string[] keys)
{
RedHitNode cur = null;
if (keys != null && keys.Length != 0)
{
// 查找父节点
cur = root;
int last = keys.Length - 1;
for (int i = 0; i < last && cur != null; i++)
{
cur = cur.Find(keys[i]);
}
// 如果找到父节点,尝试删除叶节点
if (cur != null)
{
cur = cur.Remove(keys[last]);
}
}
return cur;
}
private RedHitNode Search(string[] keys)
{
RedHitNode cur = null;
if (keys != null && keys.Length != 0)
{
// 查找节点
cur = root;
for (int i = 0; i < keys.Length && cur != null; i++)
{
cur = cur.Find(keys[i]);
}
}
return cur;
}
对于插入处理Insert,我们插入时可能和常规的插入操作有点不同。
- 根节点外的叶子节点不应该再可以加入子节点。比如“主菜单–任务–新任务1”,那么“新任务1”的节点后面不应该再有子节点,才能符合红点树的基础规则。
- 另外对于已经有的节点,再次插入失败时,直接返回原有节点,方便操作。
所以做了两个而外参数,普通插入直接调用Insert,而有额外需求的再传参限制。实现如下
private RedHitNode Insert(string[] keys)
{
return Insert(keys, false, false);
}
private RedHitNode Insert(string[] keys, bool OnlyInsertOnBranchNote, bool getNoteExist)
{
RedHitNode cur = null;
if (keys != null && keys.Length != 0)
{
// 查找节点
RedHitNode parent = root;
cur = root;
int i;
for (i = 0; i < keys.Length && cur != null; i++)
{
parent = cur;
cur = cur.Find(keys[i]);
}
if (cur == null)
{
// 节点未找到
if (!OnlyInsertOnBranchNote || cur != root || !parent.IsLeaf)
{
cur = parent;
for (i = i - 1; i < keys.Length; i++)
{
cur = cur.Add(keys[i]);
}
}
}
else
{
// 节点已存在,
if (!getNoteExist)
{
cur = null;
}
}
}
return cur;
}
这样的话,树的基础结构就实现了。后面做一下红点树的功能。
(3)树_红点处理
作为红点树,打开关闭红点,查看红点是必须功能。另外当红点信息有改变的时候,是需要播报出来的,这里单纯做一个回调在方便处理。也就是可以表现为
public class RedHitTree
{
private Action<string, int> changeNoteCallBack;
private RedHitNode root;
private char splitChar = '|';
public RedHitTree(Action<string, int> changeNoteCallBack)
{
this.root = new RedHitNode("");
this.changeNoteCallBack = changeNoteCallBack;
}
public bool HasRedHit(string redHit){}
public int GetRedHitValue(string redHit){}
public bool TurnOnRedHit(string redHit){}
public bool TurnOffRedHit(string redHit){}
private RedHitNode Insert(string[] keys){}
private RedHitNode Delete(string[] keys){}
private RedHitNode Search(string[] keys){}
}
接下来依次实现如下
public bool HasRedHit(string redHit)
{
return GetRedHitValue(redHit) > 0;
}
public int GetRedHitValue(string redHit)
{
RedHitNode node = Search(redHit.Split(splitChar));
int result = 0;
if (node != null)
{
result = node.Value;
}
return result;
}
public bool TurnOnRedHit(string redHit)
{
string[] keys = redHit.Split(splitChar);
bool flag = false;
RedHitNode node = Insert(keys, true, true);
if (node != null && node.IsLeaf)
{
if (node.Value == 0)
{
ChangeNotesValue(keys, 1);
}
flag = true;
}
return flag;
}
public bool TurnOffRedHit(string redHit)
{
string[] keys = redHit.Split(splitChar);
bool flag = false;
RedHitNode node = Search(keys);
if (node != null && node.IsLeaf && node.Value > 0)
{
if (node.Value == 1)
{
ChangeNotesValue(keys, -1);
}
flag = true;
}
return flag;
}
上面我们在更改红点数据的时候使用到了ChangeNotesValue,这个需要从根节点一直改到叶子节点,并每次更改时做一个回调。如下:
private void ChangeNotesValue(string[] keys, int value)
{
RedHitNode cur = root;
string str = "";
for (int i = 0; i < keys.Length; i++)
{
cur = cur.Find(keys[i]);
if (cur == null)
{
break;
}
cur.Value += value;
// 改变
str += keys[i];
changeNoteCallBack(str, cur.Value);
}
}
那么到这里的时候,树就完成了!!
3. 封装、检查
(1)检查
检查一下,树能否按预期正常运行
- 红点开启是否正常
TurnOnRedHit("menu|Task|newTask1"); //True
HasRedHit("menu|Task|newTask1"); //True
HasRedHit("menu|Task|newTask2"); //False
HasRedHit("menu|Task"); //True
HasRedHit("menu|Knapsack"); //False
- 子节点有一个点亮时,父节点点亮,子节点全熄灭时,父节点熄灭
TurnOnRedHit("menu|Knapsack|Prop|newProp1"); //True
TurnOnRedHit("menu|Knapsack|Prop|newProp2"); //True
HasRedHit("menu|Knapsack"); //True
TurnOnRedHit("menu|Knapsack|Prop|newProp1"); //True
HasRedHit("menu|Knapsack"); //True
TurnOnRedHit("menu|Knapsack|Prop|newProp2"); //True
HasRedHit("menu|Knapsack"); //False
- 父节点是否可以自由控制
TurnOnRedHit("menu|Task|newTask1"); //True
TurnOffRedHit("menu|Task"); //False
TurnOffRedHit("menu|Task|newTask1"); //True
ok,符合预期咯。
(2)UGF封装为组件
因为用的是UGF的框架,所以用UGF做了一下封装。
using GameFramework;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityGameFramework.Runtime;
using System;
using EventArg;
namespace GDT
{
/// <summary>
/// 自定义红点组件
/// </summary>
public class RedHitComponent : GameFrameworkComponent
{
private RedHitTree tree;
private RedHitChangeEventArgs arg;
protected override void Awake()
{
base.Awake();
tree = new RedHitTree(OnRedHitChange);
arg = new RedHitChangeEventArgs();
}
public bool HasRedHit(string redHit)
{
return tree.HasRedHit(redHit);
}
public int GetRedHitValue(string redHit)
{
return tree.GetRedHitValue(redHit);
}
public void TurnOnRedHit(string redHit)
{
if (!tree.TurnOnRedHit(redHit))
{
Debug.LogError("Failed to turn on the redHit, redHit:" + redHit);
}
}
public void TurnOffRedHit(string redHit)
{
if (!tree.TurnOffRedHit(redHit))
{
Debug.LogError("Failed to turn off the redHit, redHit:" + redHit);
}
}
public void OnRedHitChange(string key, int value)
{
arg.Key = key;
arg.Value = value;
GameEntry.Event.Fire(this, arg);
}
}
}
4. 结束咯
这样的话我们的红点系统就完成了。另外我在考虑用字典的方式来做这个红点系统,后面有做的话,可能再处理吧。