Godot游戏练习01-第15节-敌人生成动画,翻转,碰撞

今天来优化敌人, 逐步补充细节, 之前实现的敌人只会简单追踪Player, 现在给敌人添加生成动画和一些优化内容, 让敌人也更生动一些

看看效果

  1. 出生时的缩放动画
  2. 追击玩家时自动切换方向
  3. 敌人之间有碰撞, 不会完全重叠
anim1.gif

看这个左右瞅的小眼神, 竟然还有点可爱, 哈哈哈

实现过程

先修改一个小问题: 当Host主机等到倒计时为0的时候Client加入, 会产生一个错误

E 0:00:20:053   enemy_spawn_component.gd:86 @ _synchronize(): Time should be greater than zero.

在设置wait_time的时候值必须大于0, 修改timer计时同步的源码, 稍加过滤即可

@rpc("authority", "call_remote", "reliable")
func _synchronize(data: Dictionary) -> void:
    round_count = data.round_count
    var wait_time: float = data.round_timer_time_left
    if wait_time > 0:
        round_timer.wait_time = wait_time
    if data.round_timer_running:
        round_timer.start()

敌人之间的碰撞

这个最简单了

打开敌人场景enemy.tscn, 调整根节点的碰撞层级, 将层级2设置为敌人碰撞层, 在CollisionLayer中, 将Layer的2层选中, 在Mask中也选中2层

代表以下含义

  • 敌人在敌人碰撞层, 2层
  • 敌人检测来自2层的碰撞, 同时检测来自1层的碰撞(空气墙壁层)

生成动画

生成动画使用Tween实现, 这是一个制作简单动画的强大神器, 今天也算是学到一点新内容

如果希望掌握Tween的使用, 应该搜索相关资料全面学习, 特别是ease/trans组合的不同效果

func _play_spawn_animation() -> void:
    is_spawning = true
    var tween := create_tween()
    tween.tween_property(visual, "scale", Vector2.ONE, 0.4)\
        .from(Vector2.ZERO)\
        .set_ease(Tween.EASE_OUT)\
        .set_trans(Tween.TRANS_BACK)
    tween.finished.connect(func():
        is_spawning = false
    )

create_tween()是Node节点提供的内置方法, 以下是官方文档

Creates a new Tween and binds it to this node.

This is the equivalent of doing:

get_tree().create_tween().bind_node(self)
The Tween will start automatically on the next process frame or physics frame (depending on Tween.TweenProcessMode). See Tween.bind_node() for more info on Tweens bound to nodes.

Note: The method can still be used when the node is not inside SceneTree. It can fail in an unlikely case of using a custom MainLoop.

该函数相当于 get_tree().create_tween().bind_node(self)

表示创建一个tween对象, 并且绑定当前节点, 它的动画会在下一帧自动开始

绑定的效果是, 在当前节点释放时, tween对象也会自动释放, 与节点同生命周期

我们在_ready()生命周期中调用_play_spawn_animation(), 就实现了敌人出生时播放一个从小变大的动画

这里添加了一个is_spawning标志位, 表示敌人是否正在生成, 如果正在生成, 我们不希望敌人在生成过程中就开始移动

因此, process中也增加了额外判断, 仅当敌人不在生成过程中时, 才允许移动

func _process(_delta: float) -> void:
    if is_multiplayer_authority() and not is_spawning:
        if has_track_target:
            velocity = global_position.direction_to(track_target) * 40
            move_and_slide()
    if not is_spawning:
        _update_direction()

方向自动翻转

上面的_update_direction()调用就是自动翻转, 具体源码如下, 与之前Player的翻转一样, 依据目标方位在自己的左右侧来调整scale达到翻转效果

func _update_direction() -> void:
    visual.scale = Vector2.ONE\
        if track_target.x > global_position.x\
        else Vector2(-1, 1)

为了让翻转在所有peer上同步生效, 我们在process中并没有限制authority, 并且在Enemy的MultiplayerSynchronizer中新增了track_target的同步, 方便_update_direction()的效果在所有peer上一致

现在再回过头看看最开始的版本, 增加了细节之后确实看起来细致很多了

这里也引出一个问题: 我们判断敌人是否在生成中, 增加了标志位, 若以后还要判断敌人是否在攻击中, 是否有什么状态, 若不断增加标志位, 这里的逻辑将会变得稀烂

下一次我们就来学习状态管理利器: 状态机的使用

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • """1.个性化消息: 将用户的姓名存到一个变量中,并向该用户显示一条消息。显示的消息应非常简单,如“Hello ...
    她即我命阅读 4,463评论 0 6
  • 1、expected an indented block 冒号后面是要写上一定的内容的(新手容易遗忘这一点); 缩...
    庵下桃花仙阅读 954评论 1 2
  • 一、工具箱(多种工具共用一个快捷键的可同时按【Shift】加此快捷键选取)矩形、椭圆选框工具 【M】移动工具 【V...
    墨雅丫阅读 1,337评论 0 0
  • 跟随樊老师和伙伴们一起学习心理知识提升自已,已经有三个月有余了,这一段时间因为天气的原因休课,顺便整理一下之前学习...
    学习思考行动阅读 861评论 0 2
  • 一脸愤怒的她躺在了床上,好几次甩开了他抱过来的双手,到最后还坚决的翻了个身,只留给他一个冷漠的背影。 多次尝试抱她...
    海边的蓝兔子阅读 854评论 1 4

友情链接更多精彩内容