【转】Godot3游戏引擎入门之五:上下左右移动动画(上)

一、前言

前面的几篇文章陆陆续续开始介绍 2D 游戏中对玩家的一些基本操作流程了,不过功能实现非常有限,接下来我想完完整整的打造一个小 Demo :在封闭的游戏场景里控制玩家自由移动,从而达到一些简单的目标。那么, first thing first ,从解决上下左右移动功能实现开始!

上下左右移动也叫 Top-down 移动动画,这篇文章我会通过 Godot 中的节点以及相关的代码来实现玩家主角的基本移动控制。之后,再改造一下游戏场景,让我们的主角自由行走在有限的世界里。一如往常,老司机带路,如果你是编程新手,那么,前方高能请系好安全带啦!当然,前面的文章也讨论过了, GDScript 脚步非常简单,不熟悉的话可以浏览一下本系列之前的文章。

主要内容: Godot 2D 中玩家的上下左右移动及碰撞实现

阅读时间: 5 分钟

永久链接:http://liuqingwen.me/blog/2018/10/10/introduction-of-godot-3-part-5-the-basic-top-down-movement-part-1/

系列主页:http://liuqingwen.me/blog/tags/Godot/

二、正文

本篇目标

使用 AnimationPlayer 节点工具创建状态动画
使用代码控制玩家的上下左右移动功能
简单的摄像机使用和地图碰撞检测实现
通过代码实现 RigidBody2D 刚体节点的运动
创建动画

相信看了上篇文章的朋友应该对 AnimationPlayer 这个功能强大的动画工具有了一定的了解。之前只是利用它最基础的功能实现了一个简单的天鹅飞舞动画,接来下我们要使用AnimationPlayer 节点实现稍微复杂的动画制作——玩家的各种状态动画实现。

我们先创建一个场景,根节点改名为 Game ,添加两个子节点: Sprite (命名为Player )和 AnimationPlayer 节点。 Player 节点的图片材质是一张 4x5 的 SpriteSheet 精灵图集,四行分别代表下、左、右、上移动动画:

和上篇文章制作天鹅动画操作一样,分别制作四个移动动画,这四个动画都设置为循环播放,动画时长和步进大家可以自己尝试进行设置不同的时间,直到自己满意为止吧,我的就随便设置了: 时长 0.8 ,步进 0.2 ,具体设置参考上一篇文章:Godot3游戏引擎入门之四:给主角添加动画(下)。接下来才是重点:我们制作一个游戏启动时刻玩家入场动画。其实这个游戏大可不必这样做,完全是为了演示 AnimationPlayer 的强大功能,并增加一些喜感吧,当然也有一定借鉴意义,哈哈。

玩家 Player 入场动画的基本思路是这样的:主角从场景中央稍微偏上的位置快速移动到屏幕中央( position ),同时尺寸由小逐渐放大到正常缩放( scale ),并伴随透明度从完全透明完全不透明( modulate/a ),动画最后再加上一段玩家闪现的动画进行强调( visible )。哈哈,颇有主角粉墨登场的戏份啊!欢呼吧,骚年~ :sunglasses:

思路有了,关键在于使用 AnimationPlayer 来进行创建了。之前的动画制作都是一个轨道解决一个动画,但是这个动画不同了,需要一个动画实现多个属性的控制,这里就需要多个轨道了,每个属性分别创建一个轨道,然后对属性设置关键帧进行动画控制,这里需要注意的第一点是: Godot 3.1 alpha 版本中对位置和缩放属性不能直接使用钥匙?️按钮创建相应的轨道和关键帧,会重复创建轨道,这应该是一个 Bug ,不过不要紧,我们使用普通的做法,手动创建 Property Track 属性轨道,选择 Player 节点的相应属性,之后可以正常使用钥匙?️按钮创建关键帧,部分操作如下图:

上图中的勾选贝塞尔曲线过渡方式大家可以尝试一下,看看和平滑过渡有什么不同的效果吧。接下来,动画中透明度的设置是通过 Sprite 节点的 Modulate 颜色属性的Alpha(A) 通道设置实现的,至于闪现是通过控制节点的可见性 Visible 属性实现的,只要是属性, AnimationPlayer 就能创建动画,就是这么强!

最后记得把入场动画(名为 start )设置为自动播放,不要设置循环播放,毕竟主角登场了就不要重复了。

代码控制

动画制作完后的任务就交给代码来实现了!代码和上一篇文章里的左右移动代码没啥本质区别,只是多了两个方向而已,不过有两点新鲜玩意。第一个是我设置了速度变量,它是一个 Vector2 矢量,这样做的目的是:即使我们同时按住两个按键,玩家依然可以跑动或者原地踏步!大家可以体会下和上一节的不同之处。

第二个可谓是一个可以“节约生命”的功能,还记得上一节里怎么监控按键的吗?需要一个一个的常亮比如: KEY_A/KEY_LEFT 表示 A 键和左方向键。如果你是 Unity 的开发者,那么你对按键设置肯定非常熟悉,这里我不得不说 Unity 在这方面做得还是非常棒的,对键盘、操纵杆的控制设置很到位。 Godot 中同样也可以进行简化设置,比如把A 键和左方向键统一到自定义按键 left 中,具体设置在 Project Settings 中的Input Map 下添加自定义输入控制:

设置完就可以开森地写代码了:

extends Node2D

onready var player = Player onready var animationPlayer =Player

var currentAnimation = 'start'
var speed = 200

func _ready():
camera.position = player.position

func _process(delta):
var velocity = Vector2(0, 0) # 速度变量
var isMoving = false # 是否按键移动
var newAnimation = currentAnimation # 动画变量

if Input.is_action_pressed('left'):
    velocity.x += -1
    newAnimation = 'left'
    isMoving = true
if Input.is_action_pressed('right'):
    velocity.x += 1
    newAnimation = 'right'
    isMoving = true
if Input.is_action_pressed('up'):
    velocity.y += -1
    newAnimation = 'up'
    isMoving = true
if Input.is_action_pressed('down'):
    velocity.y += 1
    newAnimation = 'down'
    isMoving = true

# 速度不为0,移动玩家位置
if velocity.length() > 0:
    # 注意这里 normalize 速度矢量,否则会出现斜着走速度比单方向速度快
    player.position += velocity.normalized() * speed * delta

# 根据是否有按键按下和新动作更新动画
updateAnimation(isMoving, newAnimation)

func updateAnimation(isMoving, newAnimation):
# 如果有移动按键按下,并且改变方向,则切换动画
if isMoving and currentAnimation != newAnimation:
animationPlayer.current_animation = newAnimation
animationPlayer.play(newAnimation)
currentAnimation = newAnimation
# 未移动,但又非开始的情况,那么止移动动画
elif ! isMoving and currentAnimation != 'start':
animationPlayer.stop()
currentAnimation = 'start'
# 其他情况比如同方向继续移动,或者在开始的时候都不用处理
完成后效果:

摄像机节点

对于上面实现的效果感想如何?嗯,移动是没问题了,入场动画有,只是没有录制进来,有兴趣的朋友可以到 Github 上下载源码自己运行看看效果。不过,问题是,玩家完全可以脱离视野离家出走啊——所谓破墙而走!三种解决方式:

第一种是限制移动,让玩家在固定视窗内行动,即通过判断玩家位置坐标计算有没有超出限制范围,上一篇介绍过了
第二种是使用物理碰撞,把假的墙壁设置为真实的墙壁,这种方式下面会将
第三种是非正面解决方式,即给我们的游戏添加一个摄像机,而这个摄像机时刻跟随主角运动,那么主角就不会脱离视野了
好吧,后面两种是这篇文章的目标,对于设置摄像机,和其他游戏引擎没有区别:添加一个摄像机节点,设置一下就好了,非常简单。在 Godot 中摄像机节点是 Camera2D ,添加一个节点到游戏场景后,我们通过代码控制摄像机保持和玩家位置一致,这里唯一一个要设置的地方就是:勾选 Camera2D 的 Current 属性,激活摄像机。同时,我还稍微拉伸了镜头,使得游戏场景被放大——通过设置摄像机的 Zoom 参数实现。

上图中,最下方的文字说明了视窗属性的设置:视口模式 Mode 为 2d ,缩放模式Aspect 设置为 keep ,即保持比例,这些设置都在 Project Settings 里能找到。作用很简单,如果不设置,那么默认情况下,我们的游戏进入全屏状态后是不会进行缩放的,就像下面这样:

最后在 player.position += velocity.normalized() * speed * delta 这一句后面添加一点代码:

省略代码......

速度不为0,移动玩家位置

if velocity.length() > 0:
# 注意这里 normalize 速度矢量,否则会出现斜着走速度比单方向速度快
player.position += velocity.normalized() * speed * delta
# 更新摄像机,玩家始终在视窗内活动
updateCameraPosition()
# 省略代码......

func updateCameraPosition():
camera.position = player.position
运行游戏,查看效果:

接下来解决玩家移动无范围限制的问题。 :smiley:

添加碰撞

文章有点长,偷下懒,暂时到这里,接下来的内容放到下一节。 Stay tuned!

三、小结(上)

除了代码,这是一篇非常简单的文章,使用 AnimationPlayer 制作多个动画,以及单个动画多个轨道;使用 Camera2D 跟随玩家移动视野;设置按键规则和视窗缩放属性等。接下来的重点就交给本章的下节吧。

原文链接:https://blog.csdn.net/SpkingR/article/details/83030194

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352