Game1硬币收集 - 第1部分 玩家场景 - Godot引擎游戏开发项目实践

第1部分 - 玩家场景

您将要制作的第一个场景定义了Player对象。创建单独的玩家场景的好处之一是,您可以独立测试它,甚至在您创建游戏的其他部分之前。随着项目规模和复杂性的增加,游戏对象的这种分离将变得越来越有用。保持单个游戏对象彼此分离使得它们更容易进行故障排除,修改甚至完全替换,而不会影响游戏的其他部分。它还可以使您的Player重复使用 - 您可以将Player场景放入完全不同的游戏中,它也可以正常工作。

Player场景将显示您的角色及其动画,通过相应地移动角色来响应用户输入,并检测与游戏中其他对象的碰撞。


创建场景

首先单击Add/Create创建新节点按钮,然后选择Area2D。然后,单击其名称并将其更改为Player。单击Sence|Save Sence以保存场景。这是场景的root根节点或top-level顶级节点。您可以通过向此节点添加子项来向Player添加更多功能:

新建节点

在添加任何子项之前,最好确保不要通过单击它们来移动或调整它们的大小。选择Player节点,然后单击旁边的锁图标:

锁定子项

工具提示将说明确保对象的子项不可选,如上面的屏幕截图所示。

提示
在创建新场景时始终执行此操作是个好主意。如果主体的碰撞形状或子画面变得偏移或缩放,则可能导致意外错误并且难以修复。使用此选项,节点及其所有子节点将始终一起移动。


精灵动画

使用Area2D,您可以检测其他对象何时重叠或运行到player中,但Area2D本身没有外观,因此单击Player节点并将AnimatedSprite节点添加为子节点。 AnimatedSprite将处理player的外观和动画。请注意,节点旁边有一个警告符号。 AnimatedSprite 需要一个SpriteFrames资源,该资源包含它可以显示的动画。要创建一个,请在Inspector中找到Frames属性,然后单击<null> |New SpriteFrames

点击<null> |New SpriteFrames

接下来,在同一位置,点击<SpriteFrames>以打开SpriteFrames面板:

SpriteFrames面板

左侧是动画列表。单击默认值并将其重命名为run。然后,单击Add按钮,创建第二个名为idle的动画,第三个名为hurt

在左侧的FileSystem停靠栏中,找到run(跑),idle(待机)和hurt(受伤)的玩家图像并将其拖动到相应的动画中:

Animation Frames

每个动画的默认速度设置为每秒5帧。这有点太慢,因此请单击每个动画并将Speed(FPS)设置为8。在Inspector中,选中Playing属性旁边的On,然后选择Animation以查看动画中的动画:

设置动画精灵属性

稍后,您将编写代码以在这些动画之间进行选择,具体取决于玩家正在做什么。但首先,您需要完成设置player的节点。


碰撞形状

当使用Area2D或Godot中的其他碰撞对象之一时,它需要定义一个形状。碰撞形状定义对象占据的区域,并用于检测重叠或碰撞。形状由Shape2D定义,包括矩形,圆形,多边形和其他类型的形状。

为方便起见,当您需要向区域或物理主体添加形状时,可以将CollisionShape2D添加为子项。然后,您可以选择所需的形状类型,并可以在编辑器中编辑其大小。

添加CollisionShape2D作为Player的子节点(确保不将其添加为AnimatedSprite的子节点)。这将允许您确定玩家的hitbox或碰撞区域的边界。在Inspector中,单击Shape旁边的< null >并选择New RectangleShape2D。调整形状的大小以覆盖精灵:

拖拽碰撞形状

提示
小心不要缩放形状的轮廓!只能使用尺寸手柄(红圈)来调整形状!对于缩放的碰撞形状,碰撞将无法正常工作。

您可能已经注意到碰撞形状不是以精灵为中心。那是因为精灵本身不是垂直居中的。我们可以通过向AnimatedSprite添加一个小偏移来解决这个问题。单击节点并在Inspector中查找Offset属性。将其设置为( 0,-5 )

完成后,您的Player场景应如下所示:

Player场景

编写Player脚本

现在,您已准备好添加脚本。脚本允许您添加内置节点未提供的其他功能。单击Player节点,然后单击Add Script按钮:

添加脚本

Script Settings窗口中,您可以保留默认设置。如果您记得保存场景(请参见上面的屏幕截图),脚本将自动命名以匹配场景的名称。单击Create,您将进入脚本窗口。您的脚本将包含一些默认注释和提示。您可以删除注释(以#开头的行)。请参阅以下代码段:

extends Area2D

# class member variables go here, for example:
# var a = 2
# var b = "textvar"
func _ready():
# Called every time the node is added to the scene.
# Initialization here
pass
#func _process(delta):
# # Called every frame. Delta is time since last frame.
# # Update game logic here.
# pass

每个脚本的第一行将描述它所附加的节点类型。接下来,您将定义类变量:

extends Area2D

export (int) var speed
var velocity = Vector2()
var screensize = Vector2(480, 720)

文档
Vector2

speed变量上使用export关键字允许您在Inspector中设置其值,并让Inspector知道变量应包含哪种类型的数据。这对于您希望能够调整的值非常方便,就像调整节点的内置属性一样。单击Player节点并将Speed属性设置为350,如以下屏幕截图所示:

设置Speed速度

velocity将包含角色当前的移动速度和方向,screenize将用于设置玩家移动的极限(屏幕大小)。之后,游戏的主场景将设置此变量,但是现在您将手动设置它以便进行测试。


Player移动

接下来,您将使用_process()函数来定义player将执行的操作。每个帧都会调用_process()函数,因此您将使用它来更新您希望经常更改的游戏元素。你需要玩家做三件事:

  • 监测键盘输入
  • 沿指定方向移动
  • 播放相应的动画

首先,您需要监测输入。对于这个游戏,你有四个方向输入来监测(四个箭头键)。输入操作在Project Settings | Input Map选项卡下的项目设置中定义。在此选项卡中,您可以定义自定义事件并为其分配不同的键,鼠标操作或其他输入。默认情况下,Godot将事件分配给键盘箭头,因此您可以将它们用于此项目。

您可以使用Input.is_action_pressed()检测是否按下了输入,如果按住键则返回true,否则返回false。结合所有四个按钮的状态将为您提供合成的移动方向。例如,如果同时rightdown按住,则生成的速度矢量将为(1,1)。在这种情况下,由于我们一起添加水平和垂直移动,因此玩家移动的速度比他们刚刚水平移动的速度快。

您可以通过标准化速度来防止这种情况,这意味着将其长度设置为1,然后将其乘以所需的速度:

func get_input():
    velocity = Vector2()
    if Input.is_action_pressed("ui_left"):
        velocity.x -= 1
    if Input.is_action_pressed("ui_right"):
        velocity.x += 1
    if Input.is_action_pressed("ui_up"):
        velocity.y -= 1
    if Input.is_action_pressed("ui_down"):
        velocity.y += 1
    if velocity.length() > 0:
        velocity = velocity.normalized() * speed

文档
Input
Input is_action_pressed
Vector2 normalized()

通过在get_input()函数中将所有这些代码组合在一起,您可以更轻松地在以后更改内容。例如,您可以决定更改为模拟操纵杆或其他类型的控制器。从_process()调用此函数,然后通过生成的velocity更改player的position。为防止player离开屏幕,您可以使用clamp()函数将位置限制为最小值和最大值:

func _process(delta):

    get_input()

    position += velocity * delta

    position.x = clamp(position.x, 0, screensize.x)

    position.y = clamp(position.y, 0, screensize.y)

文档
clamp
Node._process()

单击运行Edited Scene(F6)并确认您可以在所有方向上围绕屏幕移动player。


关于delta

_process()函数包含一个名为delta的参数,然后乘以速度。什么是delta

游戏引擎尝试以每秒60帧的速度运行。但是,由于计算机速度降低,无论是在godot还是从计算机本身,这都会发生变化。如果帧速率不一致,则会影响游戏对象的移动。例如,考虑将对象设置为每帧移动10个像素。如果一切顺利进行,这将转化为在一秒钟内移动600像素。但是,如果这些帧中的某些帧需要更长时间,那么在那一秒中可能只有50帧,因此对象仅移动了500个像素。

和大多数游戏引擎和框架一样,Godot通过传递delta来解决这个问题,delta是自上一帧以来经过的时间。大多数情况下,这将是大约0.016秒(或大约16毫秒)。如果你然后采取你想要的速度(600 px / s)并乘以delta,你将获得正好10的运动。但是,如果增量增加到0.3,则对象将移动18个像素。总的来说,移动速度保持一致并且与帧速率无关。

作为附带好处,您可以以px/s为单位表示移动,而不是px/frame,这更容易可视化。


动画选择

现在玩家可以移动,你需要根据它是移动还是静止来改变AnimatedSprite正在播放的动画。run动画面向右侧,这意味着它向左移动(使用Flip H属性)应该水平翻转。将其添加到_process()函数的末尾:

    if velocity.length() > 0:
        $AnimatedSprite.animation = "run"
        $AnimatedSprite.flip_h = velocity.x < 0
    else:
        $AnimatedSprite.animation = "idle"

请注意,此代码有一点缩减。* flip_h是一个布尔属性,这意味着它可以是truefalse。布尔值也是像<*的比较结果。因此,我们可以将属性设置为等于比较结果。这一行相当于写出来像这样:

if velocity.x < 0:
    $AnimatedSprite.flip_h = true
else:
    $AnimatedSprite.flip_h = false

再次启动场景并检查动画在每种情况下是否正确。确保在AnimatedSprite中将Playing设置为On,以便播放动画。

文档
AnimatedSprite.flip_h


开始和结束玩家的运动

当游戏开始时,主场景将需要通知玩家游戏已经开始。添加start()函数如下,主场景将用于设置玩家的起始动画和位置:

func start(pos):
    set_process(true)
    position = pos
    $AnimatedSprite.animation = "idle"

当玩家遇到障碍物或时间不足时,将调用die()函数:

func die():
    $AnimatedSprite.animation = "hurt"
    set_process(false)

设置set_process(false)会导致不再为此节点调用_process()函数。这样,当玩家死亡时,他们仍然无法通过键输入来移动。


准备碰撞

玩家应该检测到它何时触碰硬币或障碍物,但是你还没有让它们这样做。没关系,因为你可以使用Godot的signal(信号)功能来使它工作。Signal是节点发送其他节点可以检测和响应的消息的一种方式。例如,当按下按钮时,许多节点都有内置信号来提醒您身体发生碰撞时。您还可以为自己的目的定义自定义信号。

通过将信号连接到您想要监听和响应的节点来使用信号。可以在Inspector或代码中进行此连接。在项目的后期,您将学习如何以两种方式连接信号。

将以下内容添加到脚本的顶部(在extends Area2D之后):

signal pickup
signal hurt

这些定义了玩家在触摸硬币或障碍物时会触发(发出)自定义信号。触摸将由Area2D本身检测到。选择Player节点,然后单击Inspector旁边的Node选项卡以查看player可以触发的信号列表:

信号列表

请注意您的自定义信号也在那里。由于其他对象也将是Area2D节点,因此您需要area_entered()信号。选择它并单击Connect。单击Connecting Signal窗口上的Connect - 您无需更改任何这些设置。 Godot会在你的脚本中自动创建一个名为_on_Player_area_entered()的新函数。

提示
连接信号时,您也可以给出要将信号链接到现有函数的名称,而不是让Godot为您创建默认函数。如果您不希望Godot为您创建功能,请将Make Function开关切换为Off

将以下代码添加到此新函数:

func _on_Player_area_entered( area ):
    if area.is_in_group("coins"):
        area.pickup()
        emit_signal("pickup")
    if area.is_in_group("obstacles"):
        emit_signal("hurt")
        die()

文档
Object.emit_signal()

当检测到另一个Area2D时,它将被传递给该函数(使用area变量)。硬币对象将具有pick()功能,该功能定义拾取时硬币的行为(例如,播放动画或声音)。当您创建硬币和障碍物时,您将它们分配给相应的组,以便可以检测到它们。

总而言之,到目前为止,这是完整的player脚本

extends Area2D

signal pickup
signal hurt

export (int) var speed
var velocity = Vector2()
var screensize = Vector2(480, 720)

func get_input():
    velocity = Vector2()
    if Input.is_action_pressed("ui_left"):
        velocity.x -= 1
    if Input.is_action_pressed("ui_right"):
        velocity.x += 1
    if Input.is_action_pressed("ui_up"):
        velocity.y -= 1
    if Input.is_action_pressed("ui_down"):
        velocity.y += 1
    if velocity.length() > 0:
        velocity = velocity.normalized() * speed

func _process(delta):
    get_input()
    position += velocity * delta
    position.x = clamp(position.x, 0, screensize.x)
    position.y = clamp(position.y, 0, screensize.y)
    if velocity.length() > 0:
        $AnimatedSprite.animation = "run"
        $AnimatedSprite.flip_h = velocity.x < 0
    else:
        $AnimatedSprite.animation = "idle"

func start(pos):
    set_process(true)
    position = pos
    $AnimatedSprite.animation = "idle"

func die():
    $AnimatedSprite.animation = "hurt"
    set_process(false)

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