在游戏中会经常遇到斜坡地形,比如众所周知的魂斗罗,角色可以在坡上移动和跳跃:

斜坡在2D游戏中很常见,处理起来也较为棘手。最初我打算用分离轴定律来实现,在建立了一个物理模型之后:

发现上坡时没什么问题,但下坡时有比较明显的弹跳,这并不是预期的结果,因为我想让物体“粘”在坡上。产生这个问题的根本原因是物体的运动分为两个阶段,

首先是水平方向的移动,最后是垂直方向的移动:

如果X轴和Y轴的速度没有一定的方式关联起来,就会产生跳跃问题。根据力的平行四边形法则,物体的运行轨迹应该像这样(以向右下坡为例):

接下来要做的工作就是计算出xspeedyspeed

为了简单起见,先将斜坡统一设置成45°,即等边直角三角形,此时只需要将Y轴速度设置为与X轴速度相同即可。

由上图可以发现,当x与y相等时,角色会贴着斜坡下降,而如果x轴的速度比y轴快,角色就会脱离斜坡,此时再加上y轴的重力加速度,将会出现上述跳跃的效果。

当然这里不是说跳跃就是错误的,有些游戏也会需要这种效果,只是在当前场景下让角色“粘”在坡上更符合需求。

任意角度的斜坡速度处理

  45°斜坡只是理想情况,实际上开发者可能要面对各种角度的斜坡,这里的“各种角度”是指小于90°,因为等于90°就相当于一面墙了。现在假设角色要在斜坡上移动的距离为distance(一般将其赋值为角色的水平速度speedX),那么其水平分量为

vx = Math.cosθ * distance;

然后令

vy = vx;

即可得出在斜坡上速度的水平和垂直分量。由于现在的水平速度是平时速度的分量,所以在斜坡上移动的速度比在平地时要慢一些。

如果不记得三角函数,可以通过下图复习一下:

上述方法都是在θ角已知的情况下展开讨论的,如果不知道θ角也没关系,它可以通过斜面的法线与垂直方向的夹角计算出来。

至此角色在下坡时跳跃的问题已经有了初步的解决方案。

现在还有一个关键问题需要解决,就是判断角色何时处于斜坡上,在斜坡上的水平移动使角色有一段时间是脱离斜坡的,必须让程序知道这个阶段的角色仍处于斜坡之上而不是离开了。

//Pseudocode
function checkCollision() {
player.onSlope = false; if(checkPlayerOnSlope()) {
player.onSlope = true;
} if(player.onSlope) {
vy = vx = Math.cosθ * distance; //计算角色在斜坡上的坐标...
}
}

为实现上面伪代码中的checkPlayerOnSlope这个方法,需要做一些额外的准备工作。

光线投射法

  光线投射法(RayCasting)是一种游戏中常见的碰撞检测手段。可以想象成一个点朝某个方向发出光线,直到光线击中或穿过待测目标。在光线投射过程中,如果记录下其穿过的Tiles,就会得到一个结果集,用这个结果集来分析判断角色所处的位置。现在假设某条光线穿过的区域如下图所示:

蓝色线段所占的区域用红色表示,但还不够精确,预期得到的结果应该像这样:

这里需要用到Bresenham画线算法(Bresenham's line algorithm)。计算机在画一条直线时,是通过像素来表现的,当像素点密集后,肉眼就看不出来了。

(图片来自wikipedia)

将该线段上所有点的坐标计算出来,观察其位于哪个Tiles内,最后就能得到线段经过的Tiles集合。

现在为角色的脚部添加3条光线:

如果其中一条经过的Tiles有斜坡,那就说明角色位于斜坡上。

斜坡物理

  当角色位于斜坡上与在平地时不一样,应该忽略重力以及其它方向上的力的影响,否则会干扰其“粘”在斜坡上。角色位于斜坡上的y坐标很容易根据下图求出:

A点的y坐标为BA'。左右移动时,根据角色的x坐标即可求出对应的y坐标slopeY,并一直令

player.y = slopeY;

直到离开斜坡。

在斜坡上忽略重力影响并不意味着角色不能进行跳跃,触发跳跃时让角色强制离地即可。

更新日志

  2017/04/09  更新角色跳跃

  2017/04/21  更新角色冲刺

  2017/05/01  更新角色状态机

  2017/05/16  更新角色攻击动画

  2017/05/22  更新角色移动攻击动画

  2017/05/24  更新角色跳跃攻击动画

  2017/06/04  更新地图绘制

  2017/06/22  更新摄像机、长距离冲刺

  2017/07/01  更新指令技

  2017/07/06  更新蓄力技

2017/07/12  更新wall jump

2017/10/13  更新斜坡地形

HTML5 2D平台游戏开发#11斜坡物理的更多相关文章

  1. HTML5 2D平台游戏开发#4状态机

    在实现了<HTML5 2D平台游戏开发——角色动作篇之冲刺>之后,我发现随着角色动作的增加,代码中的逻辑判断越来越多,铺天盖地的if() else()语句实在让我捉襟见肘: 这还仅仅是角色 ...

  2. HTML5 2D平台游戏开发#6地图绘制

    此前已经完成了一部分角色的动作,现在还缺少可以交互的地图让游戏看起来能玩.不过在开始之前应当考虑清楚使用什么类型的地图,就2D平台游戏来说,一般有两种类型的地图,Tile-based和Art-base ...

  3. HTML5 2D平台游戏开发#8指令技

    一般在动作游戏中,玩家可以通过对输入设备输入一系列的指令让角色完成某个或多个特定的动作.以格斗游戏<拳皇>为例,键入↓↘→↘↓↙← + A or C可以触发IORI的必杀技八稚女: 通过一 ...

  4. HTML5 2D平台游戏开发#7Camera

    在庞大的游戏世界中,玩家不能一览地图全貌,而是只能看到其中一部分,并一步步探索,这时就要用到一种技术来显示局部的地图,游戏术语称为摄像机(Camera).下面两张图中的白色矩形框表示了Camera的作 ...

  5. HTML5 2D平台游戏开发#5攻击

    目前为止,角色除了基本的移动外还什么都不能做,于是我打算先实现角色的攻击动画.角色的普通攻击一共可以分为三个阶段: 一段斩 二段斩 三段斩 移动攻击 跳跃攻击 触发方式为角色站立时按下J(攻击)键,角 ...

  6. HTML5 2D平台游戏开发#10Wall Jump

    这个术语不知道怎么翻译比较贴切,但并不妨碍对字面意思的理解,大概就是飞檐走壁.比如: 这是游戏<忍者龙剑传>中的场景,玩家可以通过操纵角色在墙面上移动并跳跃. 首先需要实现角色抓墙这一动作 ...

  7. HTML5 2D平台游戏开发#9蓄力技

    在很多动作游戏中,玩家操控的角色可以施放出比普通攻击更强力的蓄力技,一般操作为按住攻击键一段时间然后松开,具体效果像下面这张图: 要实现这个操作首先要记录下按键被按住的时间,初始是0: this.sa ...

  8. HTML5 2D平台游戏开发#2跳跃与二段跳

    在上一篇<Canvas制作时间与行为可控的sprite动画>中已经实现了角色的左右移动,本篇继续实现角色的一系列动作之一:跳跃.先来看看最终效果: 要实现跳跃,必须模拟垂直方向的速度和重力 ...

  9. HTML5 2D平台游戏开发#1

    在Web领域通常会用到一组sprite来展示动画,这类动画从开始到结束往往不会有用户参与,即用户很少会用控制器(例如鼠标.键盘.手柄.操作杆等输入设备)进行操作.但在游戏领域,sprite动画与控制器 ...

随机推荐

  1. 在mac上运行android的intel模拟器导致的死机问题解决

    最近用Mac开发android的时候遇到了一个烦人的问题,装上了HAXM,intel hardware accelerated execution manager以后,一运行android emula ...

  2. Makefile之“=”、":="、“+=”、“?=”

    Makefile之“=”.":=".“+=”.“?=”中几个的区别: 1.”=“符号 =表示个变量赋值: 注意: 当变量A被赋值给变量B时(B=A),这里A可以的这条指令之前定义的 ...

  3. Ubuntu14.04终端主机名+用户名修改配色方案

    首先打开终端:输入指令ls -a 然后输入指令:vi .bashrc 先按下字母A,进入编写: 在文档最后一行添加: PS1='${debian_chroot:+($debian_chroot)}\[ ...

  4. oracle数据库中函数和存储过程中的区别

    一.函数必须有返回值,过程没有返回值: 二.函数可以单独执行,过程必须通过execute执行: 三.函数可以嵌入SQL中执行,过程不能. 可以将比较复杂的查询写成函数,然后在过程中调用.

  5. P6 EPPM 安装和配置指南

    In This Section Installation and Configuration Guide Manual Installation Guides P6 Professional Inst ...

  6. [Angular] @ViewChild read custom directive and exportAs

    For example we have a component: <Card ></Card> And a driective: <Card highlighted> ...

  7. Tomcat 高性能实现关键点

    我在这里给大家讲解下Tomcat架构设计的几个关键要素,重点从性能及高可用等几个方面来讲解: 1.技术选型 (1) BIO方式适用于连接数目比较小且固定的架构,这种方式对服务器资源要求比较高,并发局限 ...

  8. 自己写的粗糙的Excel数据驱动Http接口测试框架(一)

    1.excel用例: 2.用例执行: @Testpublic void BindBank() throws Exception { String fileName = "src/main/j ...

  9. 排查 “Detected Tx Unit Hang”问题

    实现功能: 使用自己已经分配的内存让skb->data指向,而不是使用alloc_malloc(). 部分代码如下: /* * build a new sk_buff */ //struct s ...

  10. vscode - 更改emmet生成代码

    有时候生成的代码,并不适用自己,所以想想改生成代码: 因为windows查找文件/文件内容非常慢,所以借用了一下Linux的搜索命令,查找了一下 ie=edge  ,最后,找到了 expand-ful ...