打飞碟运动学与物理学结合版

架构:

在上一篇博客我介绍了运动学的打飞碟设计方法,但是在一些情况下我们希望飞碟能够在飞行过程中受到外力,比如风力或其他力的影响发生轨道的偏离。这就需要使用适配器的设计模式:

适配器模式:

适配器模式将某个类的接口转换成客户端期望的另一个接口表示,主的目的是兼容性,让原本因接口不匹配不能一起工作的两个类可以协同工作。其别名为包装器(Wrapper)。
UML图如下:
这里写图片描述
通过添加IActionManager的接口实现我们需要扩展的函数

具体实现:

承继上一篇博客,我将RoundActionManager中的addRandomAction取出作为接口函数,因为这个函数的作用在于设定飞碟的发射轨迹,因此,我只要在这个函数中获取飞碟的刚体并为之添加一个随机方向的力就可以实现飞碟的物理学飞行:

public class PhysisManager : SSActionManager, ISSActionCallback, IActionManager
{
    public RoundController scene;
    public MoveToAction action1, action2;
    public SequenceAction saction;
    float speed;
    public int If_Active;


    public void addRandomAction(GameObject gameObj)
    {
        Vector3 force = new Vector3 (
            UnityEngine.Random.Range(-10, 10),
            UnityEngine.Random.Range(-10, 10),
            UnityEngine.Random.Range(-10, 10)
        );
        int[] X = { -20, 20 };
        int[] Y = { -5, 5 };
        int[] Z = { -20, -20 };

        // 随机生成起始点和终点
        Vector3 starttPos = new Vector3(
            UnityEngine.Random.Range(-70, 70),
            UnityEngine.Random.Range(-10, 10),
            UnityEngine.Random.Range(100, 150)
        );

        gameObj.transform.position = starttPos;

        Vector3 randomTarget = new Vector3(
            X[UnityEngine.Random.Range(0, 2)],
            Y[UnityEngine.Random.Range(0, 2)],
            Z[UnityEngine.Random.Range(0, 2)]
        );
        gameObj.GetComponent<Rigidbody> ().velocity = Vector3.zero;
        gameObj.GetComponent<Rigidbody> ().AddForce (force, ForceMode.Impulse);
        MoveToAction action = MoveToAction.getAction(randomTarget, gameObj.GetComponent<DiskData>().speed);

        RunAction(gameObj, action, this);
    }

    protected  void Start()
    {
        scene = (RoundController)SSDirector.getInstance().currentScenceController;
        scene.physisManager = this;
    }

    protected new void Update()
    {
        base.Update();
    }

    public void actionDone(SSAction source)
    {
        Debug.Log("Done");
    }
}

在这之后,我们还需要在场景控制器中为这个类添加一个实例:

diskFactory = Singleton<DiskFactory>.Instance;
scoreRecorder = Singleton<ScoreRecorder>.Instance;
actionManager = Singleton<RoundActionManager>.Instance;
physisManager = Singleton<PhysisManager>.Instance;

之后,在调用这个函数的场景控制类中加上:

if(actionManager.If_Active == 1)
    actionManager.addRandomAction (disk);
else
    physisManager.addRandomAction (disk);

为了便于模式的切换,我在RoundActionManager中添加了切换的变量If_Active,当这个变量的值为1时,切换为运动学模式,否则为物理学模式。然后为飞碟的预设添加刚体,并把PhysisManager这个脚本挂载到main对象中,运行即可。
这里写图片描述

这里写图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

hellowangld

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值