2.5D游戏,角色移动限制方法。不用空气墙。

有一个项目,2.5D视角。角色在设定好的路线上自由移动,不能超出路线。

之前的做法是用空气墙,设定物理碰撞,然后角色移动。

我感觉这种做法性能有点低。手机上体验平均帧时是4ms

于是想用空间换时间,将可能的运算进行预处理,然后在运算的时候,只对部分数据进行运算以提升效率。

这个目前还在初始阶段,已经完成了移动。还未考虑到后续逻辑,不过应该没问题的。 

效果如下 

 实现思路

因为所有路线都是直线条,所以只要知道了两个点的坐标,然后基于基础菱形块的尺寸就可以算出两个块组成的一个多边形。然后在移动的时候,判断白块位置是否在这个多边形里,如果不在则不移动。

具体实现

地图节点编辑

地图里的数据,主要就是路线上的端点和拐点。所以在地图设计好了以后,就对场景里的端点和拐点创建一个节点。

节点数据初始化

创建这些节点的目的是用来组合出多边形。多边形组合如下图所示:

从图中可以看出,每一个单元,左、上、右、下都有可能会有一个相邻单元。距离可能不同。

对于我的地图,最多也只能会有4个相邻单元。

所以,可以组合出5个凸多边形。如下图所示:

 1,2,3,4个平行四边形和中间的一个菱形。

那么怎么确定,每个菱形单元附近是哪些单元呢?我用的是有意义的节点命名。比如cell_x_a_b_c_d。其中x代表了这个单元的id,后面的a、b、c、d代表了按照左上右下的顺序排布的相邻节点的id。节点管理起的作用就是用来初始化这些数据。用一个数组管理所有单元,到时候用id就可以直接拿到对应的单元数据。每一个单元数据包含了坐标,相邻单元id和单元的5个多边形。

cocos creator代码如下:

//定义数组用于存储所有单元    
private readonly worldCellsArray: Cell[] = [];
//初始化所有单元
private initWorldCellsArray() {
        for (let cellNode of this.node.children) {
            const [curCellId, ...nearCells] = cellNode.name
                .replace('cell_', '')
                .split('_')
                .map(v => Number(v));
            this.worldCellsArray.push({
                node: cellNode,
                nearCellsIdArray: nearCells,
                zoonPolygon: []
            });
        }
}
//初始化所有单元的多边形,最多5个,最少也有2个。
    private initWorldCellsPolygon() {
        const halfCellWidth = this.cellWidth / 2;
        const halfCellHeight = this.cellHeight / 2;
        this.worldCellsArray.forEach(cell => {
            const { x, y } = cell.node.position;
            cell.zoonPolygon.push([
                x - halfCellWidth,
                y,
                x,
                y - halfCellHeight,
                x + halfCellWidth,
                y,
                x,
                y + halfCellHeight
            ]);
            for (let id of cell.nearCellsIdArray) {
                const nearCell = this.worldCellsArray[id - 1];
                if (nearCell) {
                    const { x: oX, y: oY } = nearCell.node.position;
                    if (oX < x && oY < y) {
                        //left bottom
                        cell.zoonPolygon.push([
                            oX - halfCellWidth,
                            oY,
                            oX,
                            oY - halfCellHeight,
                            x,
                            y - halfCellHeight,
                            x - halfCellWidth,
                            y
                        ]);
                    } else if (oX < x && oY > y) {
                        //left top
                        cell.zoonPolygon.push([
                            oX,
                            oY + halfCellHeight,
                            oX - halfCellWidth,
                            oY,
                            x - halfCellWidth,
                            y,
                            x,
                            y + halfCellHeight
                        ]);
                    } else if (oX > x && oY > y) {
                        //right top
                        cell.zoonPolygon.push([
                            oX,
                            oY + halfCellHeight,
                            x,
                            y + halfCellHeight,
                            x + halfCellWidth,
                            y,
                            oX + halfCellWidth,
                            oY
                        ]);
                    } else if (oX > x && oY < y) {
                        //right bottom
                        cell.zoonPolygon.push([
                            x,
                            y + halfCellHeight,
                            x - halfCellWidth,
                            y,
                            oX,
                            oY - halfCellHeight,
                            oX + halfCellWidth,
                            oY
                        ]);
                    }
                }
            }
        });
    }

移动管理

当角色移动的时候,就拿到角色当前的单元id的多边形,挨个和5个多边形比较。但凡和点在一个多边形内,则返回正确,可以继续移动。并且把在哪个多边形同时也返回回去。就可以一直进行判断了。

代码如下:

    private readonly backObj: [boolean, number] = [false, 0];
    public judgeIsInCellZoonAndGetCurCellId(cellId: number, judgeVec2: Vec2) {
        const cell = this.worldCellsArray[cellId - 1];
        this.backObj[0] = false;
        this.backObj[1] = cellId;
        if (!cell) return this.backObj;
        for (let polygonIndex = 0; polygonIndex < cell.zoonPolygon.length; polygonIndex++) {
            const polygon = cell.zoonPolygon[polygonIndex];
            for (let vec2Index = 0; vec2Index < this.toolVec2Pool.length; vec2Index++) {
                this.toolVec2Pool[vec2Index].set(polygon[vec2Index * 2], polygon[vec2Index * 2 + 1]);
            }
            const isIn = Intersection2D.pointInPolygon(judgeVec2, this.toolVec2Pool);
            if (isIn) {
                this.backObj[0] = true;
                if (polygonIndex) {
                    this.backObj[1] = cell.nearCellsIdArray[polygonIndex - 1];
                }
                break;
            }
        }
        return this.backObj;
    }

因为每次都只需要进行5个多边形的判断,所以性能比用物理碰撞好得多。最后的手机预览帧时为1.5ms。还是提升了很多的。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Unity是一款流行的游戏开发引擎,能够开发各种类型的游戏,包括2.5D游戏。下面将简要介绍Unity开发2.5D游戏的教程。 首先,你需要安装Unity软件,并创建一个新项目。在项目中,你可以使用Unity的工具和资源来构建你的游戏。 其次,你需要选择一个适合的2.5D游戏场景。2.5D游戏是指使用2D的背景和角色,但是具有3D的效果和深度感。你可以选择一个平面或者地图作为游戏场景,并添加合适的纹理和贴图来增加细节和美观度。 接下来,你需要创建游戏中的角色和物体。你可以使用Unity的2D工具来绘制和设计你的角色和物体,也可以从Asset Store中下载和导入已有的角色和物体资源。 然后,你需要为角色和物体添加适当的动画和交互。Unity提供了强大的动画制作工具,通过添加关键帧和动画控制器来创建角色移动、跳跃和攻击等动作。你还可以使用Unity的C#脚本语言编写代码来实现交互逻辑,例如控制角色移动和碰撞检测等。 最后,你需要为你的游戏添加声音效果和UI界面。Unity提供了丰富的音效资源和UI工具,可以让你为游戏增添音乐、音效和用户界面等元素,提升游戏的沉浸感和用户体验。 通过以上步骤,你可以完成一个简单的2.5D游戏的开发。当然,你还可以根据自己的需求和创意进一步扩展和完善你的游戏,例如添加关卡和敌人、设计游戏规则和目标等。祝你开发一款成功的2.5D游戏

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值