当前位置: 首页 > news >正文

学习游戏制作记录(剑投掷技能)7.26

1.实现瞄准状态和接剑状态

准备好瞄准动画,投掷动画和接剑动画,并设置参数AimSword和CatchSword

投掷动画在瞄准动画后,瞄准结束后才能投掷

创建PlayerAimSwordState脚本和PlayerCatchSwordState脚本并在Player中初始化:

PlayerAimSwordState脚本:

        if(Input.GetKeyDown(KeyCode.Mouse1))//鼠标右键
{
_PlayerStateMachine.ChangeState(_Player.idleState);
}//在Update中,用于调试

PlayerGroundedState脚本:

        if(Input.GetKeyDown(KeyCode.Mouse1))
{
_PlayerStateMachine.ChangeState(_Player.AimSword);
}

为SkillManage对象创建新脚本Sword_Skil并在SkillManage脚本中初始化它:

    [Header("Skill info")]
[SerializeField] private GameObject swordPrefab;//剑的预制体
[SerializeField] private Vector2 lanchForce;//剑的发射方向
[SerializeField] private float swordGravity;//剑的重力改变

2.实现剑的投掷(剑预制体的创建与代码调用)

创建剑的对象,你可以创建一个父对象和子对象来表示一把剑,子对象负责挂载动画组件,父对象负责挂载刚体碰撞器等。

为其添加旋转和待机动画,并添加rotation参数用来控制动画

Skill脚本:

    protected Player player { get; private set; }//实例化Player,方便调用


    protected virtual void Start()
{
player = PlayerManage.instance.player;
}

为剑预制体添加Sword_Skill_Control脚本:

    private Animator anim;
private Rigidbody2D rb;
private CircleCollider2D cd;

    private Player player;

    private void Awake()
{
anim = GetComponent<Animator>();
rb = GetComponent<Rigidbody2D>(); 
cd = GetComponent<CircleCollider2D>();

}

    public void SetupSword(Vector2 _dir,float _gravityScale)//给生成的预制体一个力和重力的改变
{
rb.velocity = _dir;
rb.gravityScale = _gravityScale;
}

Sword_Skill脚本:

    public void CreateSword()
{
GameObject newSword = Instantiate(swordPrefab,player.transform.position,transform.rotation);在玩家位置创建剑对象

        Sword_Skill_Control newSkillScript =newSword.GetComponent<Sword_Skill_Control>();

        newSkillScript.SetupSword(lanchForce, swordGravity);
}

PlayerAniamtionTrigger脚本:

    private void ThrowSword()
{
SkillManage.instance.sword.CreateSword();
}//在玩家投掷动画中选择一帧添加事件来调用该函数

3.实现瞄准时方向点的生成

我们想要在玩家瞄准时随着鼠标移动展示不同的飞行轨迹并最终让剑沿着这条轨迹发射,这需要我们创建一个点的预制体,然后再游戏时生成很多个点,并套用公式让这些点排列成轨迹。

实现最终方向跟随鼠标:

Sword_Skill脚本:

private Vector2 finalDir;

    private Vector2 AnimDirection()
{
Vector2 playerPostion=player.transform.position;
Vector2 mousePosition = Camera.main.ScreenToWorldPoint(Input.mousePosition);
Vector2 direction = mousePosition - playerPostion;//由玩家指向鼠标

        return direction;
}

    protected override void  Update()
{
if(Input.GetKeyUp(KeyCode.Mouse1))
{
finalDir = new Vector2(AnimDirection().normalized.x *lanchForce.x,AnimDirection().y *lanchForce.y);//normalized是归一化处理,你可以理解把向量压缩为1,乘以我们的初始方向即最终方向
}
}

    public void CreateSword()
{
GameObject newSword = Instantiate(swordPrefab,player.transform.position,transform.rotation);

        Sword_Skill_Control newSkillScript =newSword.GetComponent<Sword_Skill_Control>();

        newSkillScript.SetupSword(finalDir, swordGravity);//调用最终方向
}

实现瞄准点的排序:

    [Header("AnimDots info")]
[SerializeField] private int numbersofDots;//生成点的数量
[SerializeField] private float spaceBetweenTwoDots;//每个点间隔的时间
[SerializeField] private GameObject dotPrefab;//点的预制体,自行创建,记得更改图层为地面
[SerializeField] private Transform dotsParent;//自行在玩家下创建一个子对象,作为生成所有点的父对象

    private GameObject[] dots;//生成点的数组

    private void GenerateDots()//生成所有点
{
dots = new GameObject[numbersofDots];
for (int i = 0; i < numbersofDots; i++)
{
dots[i]=Instantiate(dotPrefab,player.transform.position,Quaternion.identity,dotsParent);
dots[i].SetActive(false);
}
}

    protected override void Start()//调用上述函数
{
base.Start();

        GenerateDots();
}

    public void DotsActove(bool _isActive)//设置点是否可见
{
for(int i = 0; i < dots.Length; i++)//循环
{
dots[i].SetActive(_isActive);
}
}

    private Vector2 DotsPosition(float t)//返回每个点的位置
{
Vector2 position = (Vector2)player.transform.position + new Vector2(AnimDirection().normalized.x * lanchForce.x, AnimDirection().y * lanchForce.y) * t
+ .5f * (Physics2D.gravity * swordGravity) * (t * t);

//位置 = 初始位置 + 初速度 × 时间 + 0.5 × 加速度 × 时间²

        return position;
}

    protected override void  Update()
{
if(Input.GetKeyUp(KeyCode.Mouse1))
{
finalDir = new Vector2(AnimDirection().normalized.x *lanchForce.x,AnimDirection().y *lanchForce.y);
}

        if(Input.GetKey(KeyCode.Mouse1))//按住鼠标右键时时刻更新这些点的位置
{
for(int i = 0; i < dots.Length; i++)
{
dots[i].transform.position = DotsPosition(i * spaceBetweenTwoDots);
}
}
}

PlayerAimSwordState脚本:

    public override void Enter()
{
base.Enter();

        SkillManage.instance.sword.DotsActove(true);
}

    public override void Exit()
{
base.Exit();

        SkillManage.instance.sword.DotsActove(false);
}//进入瞄准时可见这些点,退出消失

演示:

http://www.lryc.cn/news/600437.html

相关文章:

  • Flutter开发实战之原生平台集成
  • 暑期算法训练.9
  • 如何查找php配置文件php.ini
  • ICMPv6报文类型详解表
  • 面条式代码(Spaghetti Code)
  • 编程与数学 03-002 计算机网络 06_网络层职责
  • RK3568笔记九十三:基于RKNN Lite的YOLOv5目标检测
  • 【Spring AI】SiliconFlow-硅基流动
  • MySQL操作进阶
  • 备份一下我的 mac mini 的环境变量配置情况
  • Android Studio Profiler工具使用流程
  • MyBatis_3
  • 零基础学后端-PHP语言(第二期-PHP基础语法)(通过php内置服务器运行php文件)
  • 【安全漏洞】防范未然:如何有效关闭不必要的HTTP请求方法,保护你的Web应用
  • Java中List集合对象去重及按属性去重
  • linux内核电源管理
  • Java同步锁性能优化:15个高效实践与深度解析
  • [spring6: Mvc-函数式编程]-源码解析
  • 栈----2.最小栈
  • 【VLLM】open-webui部署模型全流程
  • JavaWeb(苍穹外卖)--学习笔记11(Filter(过滤器) 和 Interceptor(拦截器))
  • JavaScript中.splice()的用法
  • 从压缩到加水印,如何实现一站式图片处理
  • 动态SQL标签
  • 【动态规划-斐波那契数列模型】理解动态规划:斐波那契数列的递推模型
  • 【深度之眼机器学习笔记】04-01-决策树简介、熵,04-02-条件熵及计算举例,04-03-信息增益、ID3算法
  • 深分页性能问题分析与优化实践
  • [硬件电路-94]:模拟器件 - 信号耦合,让被放大信号与静态工作点的直流偏置信号完美的融合
  • 算子推理是什么
  • Linux进程:系统运行的核心机制