Unity幸运转盘实战项目

幸运转盘主要是由两部分组成——转盘、指针,实现的方式也分两种,转盘动或者指针动,不过两者的原理都是一样的,这里就以指针动做了一个项目级的demo(由于后面的圆盘中间的那条竖线有点往左偏,所以导致那些圣诞老人的显示有些偏移)。< Demo 下载>

在项目开发中,一般这个功能的实现过程是:

1、当点击开始的时候,转盘开始动, 同时向服务端发送协议;

2、服务端返回数据(前端计算好对应的角度),当转盘旋转一定时间后开始减速直至对应角度

当然也可以等服务端返回数据后再开始模拟旋转过程,但是由于受网络的影响,这一过程可能会较长,表现效果不是很好。

关于Unity的旋转,主要有下面几类接口:Transform.Rotate(),  Transform.RotateAround(),Transform.rotation, Transform.eulerAngles。 其中 Rotate() 和 RotateAround() 都是同一类接口(里面的具体重载这里就不细说了),都是指从当前位置做指定角度的偏移,而 rotation 和 eulerAngles 都是直接赋值的字段,rotation 是一个四元数类型, eulerAngles 则是一个欧拉角。很显然,我们在做旋转的时候不需要关心具体角度,使用 Rotate() 类型做定帧偏移是最合适的,通过设置偏移的角度就能实现变速转动,比较适合转盘开始阶段的加速过程和中间的匀速过程,而当我们知道停止的具体角度时,则可已利用差值运算能精准地实现减速过程并最终停到我们需要的位置。但是有一点需要注意,Unity里面角度是逆时针计算的,而我们日常生活中一般习惯于顺时针,所以,最终的角度还需要取反一下。废话不多说,见代码:

using UnityEngine;
using UnityEngine.UI;

public class LuckyTurntable : MonoBehaviour
{
 public enum State
 {
  None,  //待机状态
  Start,  //加速阶段
  Prepared, //等待数据阶段
  End,  //减速阶段
 }

 public delegate void OnFinishCallback();
 private event OnFinishCallback OnFinish;

 /// <summary>
 /// 设置完成时的回调
 /// </summary>
 /// <param name="onFinish"></param>
 public void SetOnFinishCallback(OnFinishCallback onFinish)
 {
  OnFinish += onFinish;
 }

 /// <summary>
 /// 最大速度
 /// </summary>
 public int velocity = 3000;

 public Transform node;
 public Button btnStart;
 public Button btnStop;
 public Button btnRandom;
 public InputField input;

 private State _state;
 /// <summary>
 /// 转盘的状态
 /// </summary>
 public State CurState
 {
  get
  {
   return _state;
  }
  private set
  {
   _state = value;
   switch (value)
   {
    //不同阶段限制各按钮的点击状态
    case State.None:
     btnStart.enabled = true;
     btnStop.enabled = false;
     btnRandom.enabled = false;
     break;
    case State.Start:
     btnStart.enabled = false;
     btnStop.enabled = true;
     btnRandom.enabled = true;
     break;
    case State.Prepared:
    case State.End:
     btnStart.enabled = false;
     btnStop.enabled = false;
     btnRandom.enabled = false;
     break;
   }
  }
 }

 private float _endAngle = 0f;
 /// <summary>
 /// 最终停止的角度[0, 360]
 /// </summary>
 public float EndAngle
 {
  get
  {
   return _endAngle;
  }
  set
  {
   _endAngle = Mathf.Abs(value);
   print("End Angle: " + value);
   _endAngle = _endAngle % 360; //将角度限定在[0, 360]这个区间
   _endAngle = -_endAngle - 360 * 2; //多N圈并取反,圈数能使减速阶段变得更长,显示更自然,逼真
  }
 }

 /// <summary>
 /// 加速持续时间
 /// </summary>
 private readonly float AcceleateTime = 1f;

 /// <summary>
 /// 减速前的最短持续时间
 /// </summary>
 private float _minTime = 3.0f;
 /// <summary>
 /// 角度缓存
 /// </summary>
 private float _tmpAngle = 0f;
 /// <summary>
 /// 时间统计
 /// </summary>
 private float _time;
 /// <summary>
 /// 速度变化因子
 /// </summary>
 private float _factor;

 private void Start()
 {
  CurState = State.None;
  btnStart.onClick.AddListener(OnStartClick);
  btnStop.onClick.AddListener(OnStopClick);
  btnRandom.onClick.AddListener(OnRandomClick);
 }

 private void Update()
 {
  if (CurState == State.None)
   return;

  _time += Time.deltaTime;
  if (CurState == State.End)
  {
   //通过差值运算实现精准地旋转到指定角度(球型插值无法实现大于360°的计算)
   float k = 2f; //如果嫌减速太慢,可以加个系数修正一下
   _tmpAngle = Mathf.Lerp(_tmpAngle, EndAngle, Time.deltaTime * k);

   //这里只存在一个方向的旋转,所以不存在欧拉角万向节的问题,所以使用欧拉角和四元数直接赋值都是可以的
   node.rotation = Quaternion.Euler(0, 0, _tmpAngle);
   //node.eulerAngles = new Vector3(0, 0, _tmpAngle);

   if (1 >= Mathf.Abs(_tmpAngle - EndAngle))
   {
    CurState = State.None;
    if (null != OnFinish)
    {
     OnFinish();
     OnFinish = null;
    }
   }
  }
  else
  {
   //利用一个速度因子实现变加速的过程
   _factor = _time / AcceleateTime;
   _factor = _factor > 1 ? 1 : _factor;
   node.Rotate(Vector3.back, _factor * velocity * Time.deltaTime, Space.Self);
  }

  //当收到数据之后并且旋转了一定时间后开始减速
  if (CurState == State.Prepared && _time > _minTime)
  {
   CurState = State.End;
   _tmpAngle = GetCurClockwiseAngle();
  }
 }

 /// <summary>
 /// 将当前指针的欧拉角转换成顺时针统计角度
 /// </summary>
 /// <returns></returns>
 private float GetCurClockwiseAngle()
 {
  //由于读取到的值是[0, 180] U [-180, 0],左边由0至180递增,右边由180转变成-180,然后递增至0,所以需要转相应的转换
  return (-1) * (360 - node.eulerAngles.z) % 360;
 }

 private void OnStartClick()
 {
  CurState = State.Start;
  _time = 0;
 }

 /// <summary>
 /// 读取输入框中的角度并停止
 /// </summary>
 private void OnStopClick()
 {
  try
  {
   EndAngle = float.Parse(input.text);
  }
  catch
  {
   EndAngle = 0f;
  }
  CurState = State.Prepared;

 }

 /// <summary>
 /// 随机一个角度并停止
 /// </summary>
 private void OnRandomClick()
 {
  EndAngle = UnityEngine.Random.Range(0f, 360f);
  CurState = State.Prepared;
 }
}

功能代码基本都在上面了,如果想要Demo的话可 前往下载。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Unity实现大转盘的简单笔记

    本文实例为大家分享了Unity实现大转盘展示的具体代码,供大家参考,具体内容如下 1.unity中要实现旋转一个gameobject,我们需要改变它的transform下对应的Rotation,由于我们的大转盘是2D的视角,所以我们首先需要明确大转盘旋转的方向是旋转Rotation的Z. 2.如何实现大转盘由旋转快到慢,再到旋转指定为位置停下.查看了unity的脚本可以找到如下方法实现旋转大转盘如下: public void Rotate(Vector3 eulerAngles, Space r

  • Unity3D实现鼠标控制旋转转盘

    在培训模拟考试软件中,我们经常会遇到类似用鼠标旋转转盘打开开关的需求.让用户更加真实的操作设备仪器.接下来说下我的解决方案. 因为鼠标操作是在UI平面上,所以我们要先将被操作的模型的世界坐标转换到屏幕坐标上.代码如下: ModelScreenPos = camera.WorldToScreenPoint(Model.transform.position); 这里有个声明,这个模型代表的是转盘,而且要保证模型的中心点在转盘中心.然后我们就要计算鼠标以模型在屏幕坐标为中心点的旋转偏移量.我们开始以鼠

  • Unity 2017使用UGUI实现大转盘抽奖

    本文实例为大家分享了Unity 2017实现大转盘抽奖的具体代码,供大家参考,具体内容如下 涉及到的插件:Dotween 在"Hierarchy"面板创建一个Image,这个Image用来显示大转盘的那个圆盘,我用的UGUI里默认的那个圆圆的图片,放大后很模糊,不过不打紧 接着创建一个Panel名字改成"奖品组",这个是奖品的父物体,记得吧Panel里的Image,Canvas Renderer组件删了 然后在创建一个Panel名字改成"Awar"

  • Unity幸运转盘实战项目

    幸运转盘主要是由两部分组成--转盘.指针,实现的方式也分两种,转盘动或者指针动,不过两者的原理都是一样的,这里就以指针动做了一个项目级的demo(由于后面的圆盘中间的那条竖线有点往左偏,所以导致那些圣诞老人的显示有些偏移).< Demo 下载> 在项目开发中,一般这个功能的实现过程是: 1.当点击开始的时候,转盘开始动, 同时向服务端发送协议: 2.服务端返回数据(前端计算好对应的角度),当转盘旋转一定时间后开始减速直至对应角度 当然也可以等服务端返回数据后再开始模拟旋转过程,但是由于受网络的

  • 十个Python练手的实战项目,学会这些Python就基本没问题了(推荐)

    话不多说,直接开搞! 练习项目一:即时标记 补充utils.py: 练习项目二:画幅好画 练习项目三:万能的XML 练习项目四:新闻聚合 练习项目五:虚拟茶话会 clinet.py 来看代码: 练习项目六:使用CGI进行远程编辑 edit.cgi,用来接受index页面的名字,然后根据名字查找文件,并且输出. 最后一个文件save.cgi: 练习项目七:自定义公告板 view.py edit.py save.py 练习项目八:使用XML-RPC进行远程文件共享 client.py: 练习项目九:

  • java版飞机大战实战项目详细步骤

    本文为大家分享了java版飞机大战实战项目,供大家参考,具体内容如下 分析 飞机大战 首先对这个游戏分析,在屏幕上的物体都是飞行物,我们可以把建一个类,让其他飞行物继承这个类.游戏中应有英雄机(也就是自己控制的飞机).敌人.而敌人应该分为打死给分的飞机(就是普通飞机),另一种就是打死有奖励的敌人.他们都应该是飞行物的子类,我们也可以为普通飞机和给奖励的敌人设一个接口让他们去实现接口,这样有利于以后的扩展,我在这里给的简化版的飞机大战,主要是为了让大家了解面向对象. 第一步建立飞行物类 impor

  • java实战项目之记账软件

    本文给大家分享记账软件的实现代码,一起看看吧! 1. 项目介绍 1.1. 项目安排 1.1.1. 项目目标 本项目为JAVAEE基础班综合项目,包含了若干个知识点,达到将基础班所学知识综合使用,提高了我们对项目的理解与知识点的运用. 熟练View层.Service层.Dao层之间的方法相互调用操作.熟练dbutils操作数据库表完成增删改查通过本项目,让我们了解公司项目开发的流程,充分的掌握项目需求分析.设计与功能的代码实现.提高同学们独立分析需求与功能实现的能力. 1.1.2. 项目时间 本项

  • Unity开发VR项目问题总结分析

    目录 一.StreamVR问题: 1.运行项目时不显示手柄控制器: 2.按键动作检测出现重复问题: 3.Error during OpenVR Init: Init_InterfaceNotFound: 4.[SteamVR] Interface Not Found (105): 5.更新SteamVR时如果之前不小心删除了SteamVR的本地文件 二.Unity问题: 1.检测射线是否在UI之上(用于避免模型与UI互相干扰) 2.VS2019调试unity项目: 3.Unity 加载项目时出现

  • python游戏的魅力之冒险岛实战项目

    目录 导语 正文 效果图: 总结 导语 ​哈喽哈喽!我是木木子. 新的一天又开始了~ ​ 是的,今天是大工程,​又到了每日更新游戏环节,你们有期待吗? ​ 想了很久,这游戏估计你们也没玩过,比较少见吧!最近家里的孩子一直在沉迷<梦幻国度>. 让他见识下python的魅力,能学习也能做游戏的.不能一直玩撒~学习也很重要嘛 今天想说我们直接开始制作一款Python玛丽冒险岛游戏~ 正文 ​ ​(1)环境安装部分: Python版本:Python 3.6 开发工具:PyCharm.Python内置模

  • python游戏实战项目之童年经典超级玛丽

    导语 "超级玛丽"--有多少人还记得这款经典游戏?那个戴帽子的大胡子穿着背带裤的马里奥! 带您重温经典的回忆,超级马里奥拯救不开心!炫酷来袭. 如果您的童年,也曾被魔性的 灯~灯灯~灯~灯灯~灯洗脑~那就来怀旧一番吧! 我是华丽的分割线------------------------------ 往期写过游戏实战已经积累到30几篇了哦~嘿嘿,推一波之前的,狗头保命.jpg. 欢迎大家来领免费的游戏,开玩啦~源码可以直接找我拿滴! 在座的各位大佬,你们都玩过这里面的几个游戏吖? ​ 往期

  • python游戏实战项目之俄罗斯方块的魅力

    目录 导语 正文 效果图: ​总结 导语 ​ 为什么有这么一个简单的游戏?这个游戏如此受欢迎? 仅仅是因为它在游戏行业异常匮乏的年代出现,从而成为了一代人的记忆吗?恐怕并不是. 玩过俄罗斯方块的人都明白,它给人的感觉就像是嗑瓜子一样,一旦开始就会像上瘾一样难以停下来,绞尽脑汁只想填满空缺的地方. ​ 哈哈哈!小编每周的话基本上都会整理一些游戏代码的哈! 这一期文章就带大家来开发一款俄罗斯方块小游戏! 正文 游戏规则:由小方块组成的不同形状的板块陆续从屏幕上方落下来,玩家通过调整板块的位置和方向,

  • python游戏实战项目之智能五子棋简易版

    目录 导语 正文 总结 导语 前段时间不是制作了一款升级版本五子棋的嘛! 但是居然有粉丝私信我说: "准备拿到代码玩一下ok过去了!太难了准备放收藏夹落灰q@q~" 所噶,今天先放一个简易版本的五子棋给大家看看!学习嘛~从简单到难 还是慢慢来撒~ 学玩这篇可以学下一篇难一点的撒: Pygame实战:下五子棋吗?信不信我让你几步你也赢不了?​​​​​​​ 正文 嘿嘿!这五子棋只有人机对战了哈!不要看人机对战感觉很简单,其实代码量也超多滴. 主要代码:都有注释的撒!就不一步一步介绍了. i

  • python游戏实战项目之智能五子棋

    目录 导语 正文 总结 导语 ​ 今日游戏更新,大家好,我是木木子,每天都给大家更新最好玩的游戏!关注我,从此你再也不用去费力找游戏了! 最热门最好玩的Python游戏点击即玩! 今天推荐的游戏是五子棋小游戏--人机对战.联机对战皆可! 你是否还记得?中学时代和同桌下过的五子棋? ​ 这样下棋只怕是会被打!​!!!我怀疑他开挂了,并且找到了证据. ​ 正文 首先准备好需要的图片等素材部分如下: ​​​ 好了!直接上代码吧--设置游戏开始界面: class gameStartUI(QWidget)

随机推荐