【Unity】RPG2D龙城纷争(六)关卡编辑器之角色编辑

更新日期:2024年6月26日。

项目源码:第五章发布(正式开始游戏逻辑的章节)

索引

简介

上一篇完成的关卡编辑器已支持创建关卡环境(主要由地块单元组成),本篇,在关卡环境的基础上,需要完成角色编辑、要诀编辑等功能(角色编辑模式)。

一、角色编辑模式

1.将字段限制为只读

在开始角色编辑模式之前,我们对角色(Role)地块(Block)类定义的字段进行一些改进,为一些字段添加ReadOnly特性标记:

csharp 复制代码
    public class Role : HTBehaviour
    {
        /// <summary>
        /// 角色头像
        /// </summary>
        [Label("角色头像"), ReadOnly] public Sprite HeadImage;

		//其他省略......
	}

    public class Block : HTBehaviour
    {
        /// <summary>
        /// 类型
        /// </summary>
        [Label("类型"), ReadOnly] public BlockType Type;

		//其他省略......
	}

ReadOnly使得该字段为只读的,在检视器面板上不可编辑。

这样做的目的是防止这些属性被不小心篡改,因为他们都将交由关卡编辑器来权衡设置。

当然你也可以不这样做,只需去掉ReadOnly标记即可。

2.创建角色(刷角色)

试想一下角色编辑的功能该如何展现,第一步必然是能够创建角色,在这里我们想像刷地块一样,鼠标停留到一个位置,直接就能在该位置刷出一个角色。

那么,着手开干:

csharp 复制代码
        /// <summary>
        /// 是否激活角色刷子
        /// </summary>
        private bool _isActiveRoleBrush = false;
        /// <summary>
        /// 角色刷子类型名称
        /// </summary>
        private string[] _roleBrushTypeName = new string[] { "玩家", "敌人" };
        /// <summary>
        /// 角色刷子类型(刷出来的角色属于此阵营)
        /// </summary>
        private RoleCamp _roleBrushType = RoleCamp.Player;
        /// <summary>
        /// 角色刷子数据集(刷出来的角色使用此数据集)
        /// </summary>
        private RoleDataSet _roleDataSet;

        /// <summary>
        /// 创建一个角色
        /// </summary>
        /// <param name="block">角色所在地块</param>
        /// <param name="dataSet">角色数据集</param>
        private void CreateRole(Block block, RoleDataSet dataSet)
        {
        	//通过角色模板 _roleTmp 创建一个新角色
            GameObject obj = PrefabUtility.InstantiatePrefab(_roleTmp) as GameObject;
            obj.name = "Role";
            obj.transform.SetParent(_level.RolesRoot);
            obj.transform.localPosition = new Vector3(block.transform.position.x, block.transform.position.y, -1);
            obj.transform.localRotation = Quaternion.identity;
            obj.transform.localScale = Vector3.one;
            obj.SetActive(true);

			//为角色生成一个随机ID,并应用刷子类型(角色阵营),这里类似刷地块的逻辑
            Role role = obj.GetComponent<Role>();
            role.ID = Guid.NewGuid().ToString();
            role.Name = dataSet.name;
            role.Camp = _roleBrushType;
            //设置角色数据集
            role.SetDataSet(dataSet);
            _roles.Add(role);

			//与地块建立关联
            role.StayBlock = block;
            block.StayRole = role;

            EditorUtility.SetDirty(role);
            EditorUtility.SetDirty(block);

            Selection.activeGameObject = obj;
        }

如上,完成了创建角色的方法,再通过刷子相关的控制变量,实现UI控件面板后:

要实现按1键开刷的功能,依然是在OnSceneGui方法中补充代码:

csharp 复制代码
        private void OnSceneGui(SceneView sceneView)
        {
            if (Event.current == null)
                return;

            if (_editMode == EditMode.Map && _isActiveMapBrush)
            {
               //地块编辑模式
            }
            else if (_editMode == EditMode.Role && _isActiveRoleBrush && _roleDataSet != null)
            {
                if (Event.current.isKey && Event.current.keyCode == KeyCode.Alpha1 && Event.current.type == EventType.KeyDown)
                {
                	//将Scene视图坐标转换为世界坐标
                    Vector2 pos = ScreenToWorldPointInScene(sceneView.camera, Event.current.mousePosition);
                    //获取坐标位置的地块
                    Block block = GetBlockByPoint(pos);
                    if (block)
                    {
                    	//必须该地块不存在角色
                        if (block.StayRole == null)
                        {
                        	//才在该地块创建一个角色
                            CreateRole(block, _roleDataSet);
                        }
                    }
                }

                EditorGUIUtility.AddCursorRect(sceneView.position, MouseCursor.SlideArrow);
            }
        }

此时,我们便可以创建一个角色数据集,然后开刷了:

不过,刚刷出来的角色是没有头像的,这里显示为红色是因为他所属敌方阵营

而且,角色头像前面已经被我们搞成ReadOnly了,这里也修改不了啊(检视器面板只能看),所以,迫切需要在关卡编辑器中实现对这一个个灰色属性的编辑功能。

3.预览所有角色

首先,为了能全局预览场景中的所有角色,我们先将所有角色按阵营进行分类展示:

csharp 复制代码
        /// <summary>
        /// 玩家角色数量
        /// </summary>
        private int _playerNum;
        /// <summary>
        /// 敌人角色数量
        /// </summary>
        private int _enemyNum;
        /// <summary>
        /// 当前选中的角色物体
        /// </summary>
        private GameObject _currentSelectRoleObj;
        /// <summary>
        /// 当前选中的角色
        /// </summary>
        private Role _currentSelectRole;
        /// <summary>
        /// 是否显示所有玩家角色
        /// </summary>
        private bool _isShowPlayer = false;
        /// <summary>
        /// 是否显示所有敌人角色
        /// </summary>
        private bool _isShowEnemy = false;

通过加入上面的控制代码,然后再结合UI控件代码,实现在2个区域(玩家、敌人阵营)分别预览所有角色(UI控件代码就不贴了,看看源码就一目了然):

4.编辑选中角色属性

我们规定同时只能选中一个角色,进而进入编辑此角色状态。

那么,当选中角色时(Scene视图中选中角色物体),角色会被赋予到_currentSelectRoleObj_currentSelectRole

Tip:为了避免重复GetComponent<Role>(),使用_currentSelectRole来缓存当前角色物体身上的Role组件。

csharp 复制代码
        private void EditRoleGUI()
        {
        		//如果Scene视图中选择的目标物体改变
                if (_currentSelectRoleObj != Selection.activeGameObject)
                {
                	//则尝试获取其上的Role组件
                    _currentSelectRoleObj = Selection.activeGameObject;
                    _currentSelectRole = _currentSelectRoleObj != null ? _currentSelectRoleObj.GetComponent<Role>() : null;
                }
                if (_currentSelectRole != null)
                {
                	//此时便选中了角色,在这里展示角色的相关属性,同时支持编辑
                }
        }

通过敲完繁琐的UI控件代码后,现在的编辑器界面便是这样:

在这里,我们可以重新赋予角色的数据集,以更换其内核。

赋予头像灰色头像(仅当角色禁用时展示),头像改变后,会立即体现在角色头像渲染器上:

角色的ID属性极其重要,在关卡间角色的属性继承存档读档剧情对话等一系列需要定位指定角色的功能,都是通过ID来确定的,所以ID不能重复,当然这里默认生成的Guid.NewGuid()是绝对不重复的。

只不过,为了方便后续关卡进行对应,主角的ID建议单独设置,比如某个主角ID为001,那么在所有关卡中,他的ID都必须为001

当我们把角色状态切换为Not Yet On Stage时,此角色将延时登场:

然后,下面列出了角色的8个要诀栏位对应的数据集,我们可以创建一系列要诀数据集,然后给每个角色都进行配置,4-8栏位会自动根据角色的等级进行激活,当然,也可以在后期使用要诀研习系统,为指定的角色学习任意要诀。

5.移动角色位置

如果我们想修改一个角色的位置,直接删了重新刷是一种笨办法,但因为角色的位置与其所处的地块存在直接关联,我们手动调节角色的位置自然是不可行的,所以需要实现控制角色移动的功能,比如:

csharp 复制代码
        if (GUILayout.Button("上移", "ButtonLeft"))
        {
        	//保留角色所站的旧地块
            Block oldBlock = _currentSelectRole.StayBlock;
            //获取角色位置上方+1格的地块,为新地块
            Block newBlock = GetBlockByIndex(oldBlock.Pos + new Vector2Int(0, 1));
            if (newBlock != null && newBlock.StayRole == null)
            {
                _currentSelectRole.transform.localPosition = new Vector3(newBlock.transform.position.x, newBlock.transform.position.y, -1);

				//交换新、旧地块的关联属性
                oldBlock.StayRole = null;
                _currentSelectRole.StayBlock = newBlock;
                newBlock.StayRole = _currentSelectRole;

                EditorUtility.SetDirty(oldBlock);
                EditorUtility.SetDirty(newBlock);
                EditorUtility.SetDirty(_currentSelectRole);
            }
        }

需要上、下、左、右移动功能,才能让角色可以变换到地图中的任意位置:

6.移除角色

同样的,自然需要移除角色的功能,我们在UI面板上已经画好了控件,只需实现对应的移除角色功能即可:

csharp 复制代码
        /// <summary>
        /// 移除一个角色
        /// </summary>
        /// <param name="role">角色</param>
        private void RemoveRole(Role role)
        {
            _roles.Remove(role);

            Block block = role.StayBlock;
            role.StayBlock = null;
            block.StayRole = null;

			//立即销毁角色物体
            DestroyImmediate(role.gameObject);
            EditorUtility.SetDirty(block);
        }

至此,角色编辑的功能便初步完成了,我们能够在关卡上任意布局角色编辑角色属性编辑角色要诀等,为后续驱动角色,进而驱动整个游戏逻辑打下了坚实的基础。

相关推荐
哭哭啼32 分钟前
VSCode中常用的快捷键
ide·vscode·编辑器
mit6.82410 小时前
ONLYOFFICE 8.1 桌面编辑器测评:引领数字化办公新潮流
前端·笔记·编辑器
网路侠客12 小时前
您的私人办公室!-----ONLYOFFICE8.1版本的桌面编辑器测评
编辑器
阿林学习计算机12 小时前
【工具测评】ONLYOFFICE8.1版本桌面编辑器测评:好用!
编辑器
搂……住13 小时前
ONLYOFFICE8.1版本桌面编辑器测评
编辑器
乐吾乐科技13 小时前
【国产开源可视化引擎Meta2d.js】图纸
开发语言·前端·javascript·编辑器·web·数据可视化
乐吾乐科技14 小时前
【国产开源可视化引擎】Meta2d.js简介
前端·编辑器·流程图·web·数据可视化·大屏端
不会C语言的男孩14 小时前
vim常用操作
linux·编辑器·vim
拿我格子衫来15 小时前
图形编辑器基于Paper.js教程04: Paper.js中的基础知识
图像处理·编辑器·几何学
weixin_5051544615 小时前
易用且免费的在线3D交互编辑器?
3d·编辑器·数字孪生·数据可视化·3d产品配置器·3d交互展示