教你用js开发【风来之国】游戏之地图篇!入门教程,大佬勿进~

简介: 毕竟这是一个开发者社区,本文希望借这款游戏来教大家如何学习游戏开发。

本系列教程大概会有三篇


  • 地图篇:基于图像的路墙系统(GraphicBased)


  • 移动篇:角色和地图相对移动


  • 互动篇:和地图上的元素互动


会不会写完看大家的反响:p


地图篇


地图决定了玩家所操控的游戏角色可以移动的范围和区域,游戏开发中有多种方式来实现地图。但无论是3D游戏,还是2D游戏,地图系统的底层一般都是基于2d图形。以风来之国游戏中下面这个场景为例


image.png


红色画笔是玩家操作角色允许移动的区域,我们要如何去实现呢?首先先把这个场景地图分层


  • 地板层角色永远在地板之上


  • 建筑层会根据角色的位置相互覆盖


因为我没有风来之国的美术资源


所以我这个灵堂级画师....


手绘了一个游戏地图


image.png


和原版可谓是毫无分别


搭架子


这里我们会使用 PIXI.js 引擎来搭建这个游戏系统,需要注意的是,我希望大家明白,文章重点在说原理,不是非得要用某个指定的引擎。


下面的代码可让玩家控制上下左右移动。


<script src="https://cdnjs.cloudflare.com/ajax/libs/pixi.js/5.1.3/pixi.min.js"></script>
<script type="text/javascript">
        //初始化PIXI
        const app = new PIXI.Application({
                backgroundColor:0xffffff
        });
        window.onload=function(){
                //将PIXI的渲染视图添加到网页body中
                document.body.appendChild(app.view);
                app.renderer.autoResize = true;
                app.renderer.resize(window.innerWidth, window.innerHeight);
                //加载图片资源
                app.loader.
                add('player', 'player.png').
                add('floor','floor.jpg').
                load((loader, resources) => {
                    //将图片资源添加到场景
                    const floor = new PIXI.Sprite(resources.floor.texture);
                    app.stage.addChild(floor);
                    const player = new PIXI.Sprite(resources.player.texture);
                    app.stage.addChild(player);
                    //将玩家图形的原点设置为中心点
                    player.anchor.x = 0.5;
                    player.anchor.y = 0.5;
                    app.ticker.add(() => {
                        //帧同步事件的回调
                    });
                    //上下左右键按下后角色移动
                    window.onkeydown=function (e) {
                            if(e.keyCode==37) {
                                 player.x-=2;
                            }else if(e.keyCode==39){
                                player.x+=2;
                            }
                            if(e.keyCode==38){
                                player.y-=2;
                            }else if(e.keyCode==40){
                                player.y+=2;
                            }
                        }
                });
        }
</script>


image.png


TADA! 现在我们的角色动起来了


现在这个移动不是很流畅,并且无法响应两个按键同时按下,也就是无法进行8向移动(↖↑↗ ←→ ↙↓↘),我们如此调整一下代码


app.ticker.add(() => {
    var nextX = player.x+moveDelta.x;
    var nextY = player.y+moveDelta.y;
    player.x=nextX;
    player.y=nextY;
});
window.onkeyup=function(e){
    if(e.keyCode==37||e.keyCode==39) {
        moveDelta.x=0;
    }else if(e.keyCode==38||e.keyCode==40){
        moveDelta.y=0;
    }
}
window.onkeydown=function (e) {
    if(e.keyCode==37) {
        moveDelta.x=-2;
    }else if(e.keyCode==39){
        moveDelta.x=2;
    }
    if(e.keyCode==38){
        moveDelta.y=-2;
    }else if(e.keyCode==40){
        moveDelta.y=2;
    }
}


image.png


丝滑多了,并且是8向移动的!


路墙系统


有些地方角色能去,有些地方不能去,这个要如何实现呢?在大部分游戏中,大致可以分为两种实现方式,其一是基于瓦片来实现,就是把地图分为不同的格子,用二维数组来决定哪些地方是路,哪些地方是墙,比如下面这个二维数组,1代表墙,0代表路。如果用相同尺寸的地板素材,可以很方便的建立路墙系统。


roadarea=[
    1,1,1,1,1,1,1,1,
    1,0,0,0,0,0,0,1,
    1,0,0,0,0,0,0,1,
    1,1,1,1,1,1,1,1
]


但这种方式比较适合由固定尺寸的瓦片图拼接的游戏,我将另起一系列文章来分享瓦片游戏系统的搭建。在本文中,我们使用基于图像的路墙系统GraphicBased。首先我们需要一个路墙区域标识图,大家注意看下图,这里不能行走的区域用红色来表示。


image.png


相信大家已经明白我们要怎么做了!


我们只需要预判移动的目标位置下的像素是否为红色就不让继续移动...


下面我们来看看代码,注意,为了代码不过于长,这里贴的代码都是片段,要仔细阅读注释来理解或者对照我提供的源码


//将上图中的路墙标识图加载到系统中 
add('roadarea','roadarea.jpg')
const roadarea = new PIXI.Sprite(resources.roadarea.texture);
app.stage.addChild(roadarea);
//获得路墙标识图的全部像素数组
//格式[r,g,b,a],所以数组长度为路墙标识图的 宽x高x4
roadareaPixels = app.renderer.extract.pixels(roadarea);
//判断x,y坐标下的像素是否为墙体的标识颜色
function isHitWall(x,y){
    var width = 1728;//路墙标识图宽度
    var position = (width * y + x) * 4;//还原像素序号
    //取出颜色值r,g,b,a
    var r=roadareaPixels[position],g=roadareaPixels[position+1],b=roadareaPixels[position+2],a=roadareaPixels[position+3];
    //jpeg格式输出后颜色会略有损失,这里根据实际调整
    //就是判断x,y坐标下的颜色是否为红色
    if(r==255&&g<50){
            console.log("hit the wall");
            return true;
    }else{
            return false;
    } 
}
app.ticker.add(() => {
    var nextX = player.x+moveDelta.x;
    var nextY = player.y+moveDelta.y;
    //只有在移动的目标坐标下不是墙体颜色时才允许移动 
    if(!isHitWall(player.x,nextY)){
        player.y=nextY;
    }
    if(!isHitWall(nextX,player.y)){
        player.x=nextX;
    }
});


大家仔细看GIF,遇到墙时,调试信息输出了hit the wall


image.png


这部分代码有看不到明白的,可以仔细看看这篇文章《# 使用Javascript制作BadApple字符画视频》,其中使用的技术原理是一样的。


细节优化


现在判定的x,y来自角色的中心点,我们把角色的半径算上。并且隐藏路墙标识图,我们不需要看见它。


// app.stage.addChild(roadarea);
//在判定移动的目标位置时加上操控角色的半径,这里差不多是45
if(!isHitWall(player.x,nextY+45*moveDelta.y)){
    player.y=nextY;
}
if(!isHitWall(nextX+45*moveDelta.x,player.y)){
    player.x=nextX;
}


image.png


本文就此结束,下一篇我将给大家讲讲


  • 移动篇:角色和地图相对移动
相关文章
|
9天前
|
Web App开发 JavaScript 前端开发
Node.js 是一种基于 Chrome V8 引擎的后端开发技术,以其高效、灵活著称。本文将介绍 Node.js 的基础概念
Node.js 是一种基于 Chrome V8 引擎的后端开发技术,以其高效、灵活著称。本文将介绍 Node.js 的基础概念,包括事件驱动、单线程模型和模块系统;探讨其安装配置、核心模块使用、实战应用如搭建 Web 服务器、文件操作及实时通信;分析项目结构与开发流程,讨论其优势与挑战,并通过案例展示 Node.js 在实际项目中的应用,旨在帮助开发者更好地掌握这一强大工具。
30 1
|
2月前
|
小程序 JavaScript 前端开发
uni-app开发微信小程序:四大解决方案,轻松应对主包与vendor.js过大打包难题
uni-app开发微信小程序:四大解决方案,轻松应对主包与vendor.js过大打包难题
691 1
|
2月前
|
JavaScript 前端开发 安全
TypeScript的优势与实践:提升JavaScript开发效率
【10月更文挑战第8天】TypeScript的优势与实践:提升JavaScript开发效率
|
2月前
|
JavaScript 前端开发 IDE
深入理解TypeScript:提升JavaScript开发的利器
【10月更文挑战第8天】 深入理解TypeScript:提升JavaScript开发的利器
31 0
|
9天前
|
开发框架 JavaScript 前端开发
TypeScript 是一种静态类型的编程语言,它扩展了 JavaScript,为 Web 开发带来了强大的类型系统、组件化开发支持、与主流框架的无缝集成、大型项目管理能力和提升开发体验等多方面优势
TypeScript 是一种静态类型的编程语言,它扩展了 JavaScript,为 Web 开发带来了强大的类型系统、组件化开发支持、与主流框架的无缝集成、大型项目管理能力和提升开发体验等多方面优势。通过明确的类型定义,TypeScript 能够在编码阶段发现潜在错误,提高代码质量;支持组件的清晰定义与复用,增强代码的可维护性;与 React、Vue 等框架结合,提供更佳的开发体验;适用于大型项目,优化代码结构和性能。随着 Web 技术的发展,TypeScript 的应用前景广阔,将继续引领 Web 开发的新趋势。
25 2
|
15天前
|
JavaScript 前端开发 测试技术
探索现代JavaScript开发的最佳实践
本文探讨了现代JavaScript开发中的最佳实践,涵盖ES6+特性、现代框架使用、模块化与代码分割、测试驱动开发、代码质量与性能优化、异步编程、SPA与MPA架构选择、服务端渲染和静态站点生成等内容,旨在帮助开发者提升代码质量和开发效率。
|
24天前
|
Web App开发 存储 JavaScript
深入浅出Node.js后端开发
【10月更文挑战第31天】本文将引导你进入Node.js的奇妙世界,探索其如何革新后端开发。通过浅显易懂的语言和实际代码示例,我们将一起学习Node.js的核心概念、搭建开发环境,以及实现一个简单但完整的Web应用。无论你是编程新手还是希望拓展技术的开发者,这篇文章都将为你打开一扇通往高效后端开发的大门。
|
19天前
|
Web App开发 JavaScript 前端开发
深入浅出Node.js后端开发
【10月更文挑战第36天】本文将引导您探索Node.js的世界,通过实际案例揭示其背后的原理和实践方法。从基础的安装到高级的异步处理,我们将一起构建一个简单的后端服务,并讨论如何优化性能。无论您是新手还是有经验的开发者,这篇文章都将为您提供新的视角和深入的理解。
|
20天前
|
运维 监控 JavaScript
鸿蒙next版开发:分析JS Crash(进程崩溃)
在HarmonyOS 5.0中,JS Crash指未处理的JavaScript异常导致应用意外退出。本文详细介绍如何分析JS Crash,包括异常捕获、日志分析和典型案例,帮助开发者定位问题、修复错误,提升应用稳定性。通过DevEco Studio收集日志,结合HiChecker工具,有效解决JS Crash问题。
38 4
|
25天前
|
Web App开发 JavaScript 前端开发
深入浅出Node.js后端开发
【10月更文挑战第30天】本文将通过一个Node.js的简单示例,引导你进入Node.js的世界。我们将从基础概念讲起,然后一步步深入到代码实现,最后总结Node.js在后端开发中的优势和应用场景。无论你是前端开发者还是后端新手,这篇文章都将为你打开一扇了解Node.js的大门。
46 2