【游戏开发实战】Unity实现类似GitHub地球射线的效果(LineRenderer | 贝塞尔曲线)

2023-05-16

文章目录

      • 一、前言
      • 二、实现思路
        • 1、曲线的本质是什么?
        • 2、如何绘制曲线?
        • 3、如何构造曲线的点?
        • 4、如何在球的表面选取两个点?
        • 5、如何让曲线有动画效果?
      • 三、具体实操
        • 1、创建Unity工程
        • 2、制作宇宙天空盒
          • 2.1、天空盒贴图
          • 2.2、天空盒材质球
          • 2.3、设置场景天空盒
        • 3、制作地球
          • 3.1、创建球体
          • 3.2、地球贴图
          • 3.3、制作地球材质球
          • 3.4、制作云层
        • 4、制作LineRenderer
          • 4.1、创建LineRenderer
          • 4.2、调节宽度
          • 4.3、设置材质球
        • 5、Line脚本:曲线逻辑
        • 6、Earth脚本:地球逻辑
        • 7、运行效果
      • 四、动态效果
        • 1、Line脚本:动态效果
        • 2、运行效果
      • 五、加点粒子特效
      • 六、工程源码
      • 七、完毕

一、前言

嗨,大家伙,我是新发。
好久不见,这是2022年第一篇博客,今天有同学私信我,问我在Unity中如何实现这种地球辐射线的效果,
在这里插入图片描述
这一看,我就想到了GitHub主页的地球射线,
请添加图片描述
那么,今天就来讲讲如何实现这个效果吧~
本文最终效果如下:
请添加图片描述
本文工程源码见文章末尾~

二、实现思路

我们先把问题进行拆解,
在这里插入图片描述

现在挨个问题进行思考与解答。

1、曲线的本质是什么?

一条曲线,它本质上是由N条直线段组成的,当N足够大的时候,曲线就会看起来很平滑,我用Blender给大家演示一下,请添加图片描述

2、如何绘制曲线?

我在之前的一些篇文章中有讲过使用LineRenderer来绘制曲线,这里我们是用LineRenderer来实现就好啦。
往期相关文章:
《【游戏开发解答】教你在Unity中使用LineRenderer制作行军蚂蚁线(行军 | 虚线 | 路径 | 线段)》

《【游戏开发实战】Unity实现水果忍者切水果的刀痕效果教程(两种实现方式:TrailRenderer、LineRenderer)》

《【游戏开发实战】TapTap物理画线游戏,教你使用Unity实现2D物理画线功能》

3、如何构造曲线的点?

上面我们说使用LineRenderer来绘制曲线,而LineRenderer需要我们告诉它点的坐标,那么我们如何来构造曲线的点坐标呢?常用的曲线有B样条、贝塞尔曲线等,关于贝塞尔曲线,我之前也有专门写过文章,
《【游戏开发进阶】玩转贝塞尔曲线,教你在Unity中画Bezier贝塞尔曲线(二阶、三阶),手把手教你推导公式》

这里我们就用贝塞尔曲线即可,这里我打算使用三阶贝塞尔曲线,三阶贝塞尔曲线需要知道四个点坐标,现在问题变成了我们如何确定这四个点的坐标,其中起始点和终止点是在球的表面选取,中间两个点我们通过一些几何运算来获得,现在问题变成了如何在球的表面获取两个点。

4、如何在球的表面选取两个点?

这是一个几何问题,我们已知球的球心坐标,想要在球的表面随机选取两个点,我们只需要在球心处随机两个方向向量,然后从球心出发,分别沿着这两个方向走一个半径长度的距离即可到达球的表面,画个图方便大家理解,
在这里插入图片描述
写成代码大概是这样子:

// 球心坐标
Vector3 centerPos = Vector3.zero;
// 球半径
float radius = 1;
// 随机一个单位向量作为方向向量
Vector3 randomDir = new Vector3(Random.Range(-1f, 1f), 
						Random.Range(-1f, 1f), 
						Random.Range(-1f, 1f)).normalized;
// 球表面的点				
Vector pos = centerPos + randomDir * radius;

我们使用上面的方法分别取到球表面的两个点即可。

上面我们说使用三阶贝塞尔曲线,现在还差中间两个控制点的坐标,我们可以用下面这样的方法来计算控制点的坐标:先求出球表面两个点的连线的中点,然后从球心指向这个中点,得到一个方向向量,再分别从点1点2朝这这个方向向量走一段距离,得到控制点1控制点2的坐标,如下
在这里插入图片描述

5、如何让曲线有动画效果?

我们可以看到曲线是有一个从起始点飞向目标点的效果的,
请添加图片描述
这个我们可以在脚本中动态设置LineRenderer的点来达到这个效果。

三、具体实操

讲完了理论,下面我们就来具体实操吧~

1、创建Unity工程

工程名就叫UnityEarthRay好了,工程模板使用3D,点击创建,
在这里插入图片描述

2、制作宇宙天空盒

2.1、天空盒贴图

我们先去找6张宇宙天空盒的图片,导入到工程目录中,如下
在这里插入图片描述

2.2、天空盒材质球

创建一个材质球,重命名为Skybox,如下
在这里插入图片描述

设置材质球的ShaderSkybox/6 Sided,并设置六个面的贴图,如下,
在这里插入图片描述

2.3、设置场景天空盒

Skybox材质球拖入Scene窗口中,请添加图片描述
我们也可以通过点击菜单Window / Rendering / Lighting
在这里插入图片描述
然后点击Environment标签页,设置Skybox Material来设置天空盒,
在这里插入图片描述
如果想改回原来的天空盒,只需把Skybox Material设置为默认的天空盒材质即可,如下
请添加图片描述

3、制作地球

3.1、创建球体

Hierarchy窗口空白处右键鼠标,点击菜单3D Object / Sphere,创建一个球体,
在这里插入图片描述
重命名为Earth,作为地球的外形,
在这里插入图片描述
如下
在这里插入图片描述

3.2、地球贴图

去找一下地球的贴图,导入工程目录中,我找了云层贴图、地球贴图、灯光贴图,如下
在这里插入图片描述

3.3、制作地球材质球

创建一个材质球,重命名为Earth
在这里插入图片描述
设置材质球的ShaderStandard,设置Albedo贴图,调节Matallic(金属度)和Smoothness(光滑度),开启Emission(自发光),并设置发光贴图,设置发光颜色为橘黄色,如下
在这里插入图片描述

然后把材质球赋值给模型,如下
在这里插入图片描述
此时效果,
在这里插入图片描述

3.4、制作云层

同理,我们在Earth子节点下再创建一个球体,重命名为Clouds,如下

在这里插入图片描述
调节Clouds的缩放,让它比Earth大一点点,
在这里插入图片描述

创建一个材质球,重命名为Clouds
在这里插入图片描述
设置材质球的ShaderStandard,设置Rendering ModeTransparent(透明),设置Albedo贴图,设置颜色值的A通道为0,调节Matallic(金属度)和Smoothness(光滑度),开启Emission(自发光),并设置发光贴图,设置发光颜色为灰色,如下
在这里插入图片描述
Clouds材质球赋值给Clouds节点,
在这里插入图片描述
此时效果
请添加图片描述

4、制作LineRenderer

4.1、创建LineRenderer

Earth节点右键鼠标,点击菜单Effects / Line,创建一个LineRenderer
在这里插入图片描述
如下
在这里插入图片描述
此时可以看到场景中多了一条粗粗的白色短线,它就是LineRenderer本君了,
在这里插入图片描述

4.2、调节宽度

我们可以调节一下它的宽度,让它细一点,
在这里插入图片描述
如下
在这里插入图片描述

4.3、设置材质球

我们可以给他创建一个材质球,
在这里插入图片描述
如下
在这里插入图片描述

5、Line脚本:曲线逻辑

创建一个C#脚本,重命名为Line,编写曲线的逻辑代码,
在这里插入图片描述
首先我们封装一个三阶贝塞尔曲线的函数,如下

注:这里三阶贝塞尔曲线代码看不懂的建议先看我之前的这篇文章,《【游戏开发进阶】玩转贝塞尔曲线,教你在Unity中画Bezier贝塞尔曲线(二阶、三阶),手把手教你推导公式》

// Line.cs

// 三阶贝塞尔曲线
private Vector3 cubicBezier(Vector3 a, Vector3 b, Vector3 c, Vector3 d, float t)
{
    Vector3 aa = a + (b - a) * t;
    Vector3 bb = b + (c - b) * t;
    Vector3 cc = c + (d - c) * t;

    Vector3 aaa = aa + (bb - aa) * t;
    Vector3 bbb = bb + (cc - bb) * t;
    return aaa + (bbb - aaa) * t;
}

接着我们定义曲线的最大点数,

/// <summary>
/// 曲线最大点数
/// </summary>
public int MAX_FRAG_CNT = 100;

声明一个List用于存储点坐标

private List<Vector3> posList = new List<Vector3>();

我们声明一个lineRenderer成员,并在Awake中获取它,

[RequireComponent(typeof(LineRenderer))]
public class RadiationLine : MonoBehaviour
{
    private LineRenderer lineRenderer;
	...
	
	void Awake()
    {
        lineRenderer = GetComponent<LineRenderer>();
    }
	
	...
}

封装一个绘制曲线的方法,如下

// Line.cs

/// <summary>
/// 绘制曲线
/// </summary>
/// <param name="fromPos">起始坐标</param>
/// <param name="ctrlPoint1">控制点1</param>
/// <param name="ctrlPoint2">控制点2</param>
/// <param name="toPos">目标坐标</param>
public void DrawRay(Vector3 fromPos, Vector3 ctrlPoint1, Vector3 ctrlPoint2, Vector3 toPos)
{
    posList.Clear();
    for (int i = 0; i <= MAX_FRAG_CNT; ++i)
    {
        posList.Add(cubicBezier(fromPos, ctrlPoint1, ctrlPoint2, toPos, (float)i / MAX_FRAG_CNT));
        lineRenderer.positionCount = posList.Count;
    }
    lineRenderer.SetPositions(posList.ToArray());
}

Line脚本挂到Line节点上,如下
在这里插入图片描述

6、Earth脚本:地球逻辑

创建一个C#脚本,重命名为Earth,编写地球逻辑代码,
在这里插入图片描述
代码很简单,这里就不拆开讲解了,我都有写注释,

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

/// <summary>
/// 地球逻辑
/// </summary>
public class Earth : MonoBehaviour
{
    private Transform selfTrans;
    // 曲线
    public Line line;


    void Start()
    {
        selfTrans = transform;
        StartCoroutine(FireLine());
    }

    void Update()
    {
        // 地球自转
        selfTrans.Rotate(Vector3.up * Time.deltaTime, Space.Self);
    }

    /// <summary>
    /// 发射辐射射线
    /// </summary>
    /// <returns></returns>
    IEnumerator FireLine()
    {
        this.line.gameObject.SetActive(false);
        while (true)
        {
            // 循环生成曲线,这里只是演示效果,我就不是用对象池了
            var line = Instantiate(this.line);
            line.gameObject.SetActive(true);
            line.transform.SetParent(selfTrans);

            // 半径
            var radius = selfTrans.localScale.x / 2f;

            // 在地球表面随机一个起始点
            var from = selfTrans.position + new Vector3(Random.Range(-1f, 1f), Random.Range(-1f, 1f), Random.Range(-1f, 1f)).normalized * radius;

            // 在地球表面随机一个终点
            var to = selfTrans.position + new Vector3(Random.Range(-1f, 1f), Random.Range(-1f, 1f), Random.Range(-1f, 1f)).normalized * radius;
            var center = (from + to) / 2f;
            // 控制点1
            var ctrlPoint1 = from + (center - selfTrans.position).normalized * (from - to).magnitude * 0.6f;
            // 控制点2
            var ctrlPoint2 = to + (center - selfTrans.position).normalized * (from - to).magnitude * 0.6f;

            line.DrawRay(from, ctrlPoint1, ctrlPoint2, to);
            // 随机一个时间后销毁曲线
            Destroy(line.gameObject, Random.Range(4, 7));
            // 随机等待一个事件
            yield return new WaitForSeconds(Random.Range(0.3f, 2f));
        }
    }
}

Earth脚本挂到Earth节点上,并赋值Line成员,如下
在这里插入图片描述

7、运行效果

运行Unity,基本的效果已经出来了,
请添加图片描述

四、动态效果

1、Line脚本:动态效果

我们要让曲线有动态的效果,把原来的DrawRay改成协程,动态设置点的坐标,如下,

// Line.cs

/// <summary>
/// 绘制曲线
/// </summary>
/// <param name="fromPos">起始坐标</param>
/// <param name="ctrlPoint1">控制点1</param>
/// <param name="ctrlPoint2">控制点2</param>
/// <param name="toPos">目标坐标</param>
public IEnumerator DrawRay(Vector3 fromPos, Vector3 handPos1, Vector3 handPos2, Vector3 toPos)
{
    for (int i = 0; i <= MAX_FRAG_CNT; ++i)
    {
        posList.Clear();
        for (int j = 0; j <= i; ++j)
        {
            posList.Add(cubicBezier(fromPos, handPos1, handPos2, toPos, (float)j / MAX_FRAG_CNT));
        }
        lineRenderer.positionCount = posList.Count;
        lineRenderer.SetPositions(posList.ToArray());
        yield return new WaitForSeconds(0.02f);
    }

    yield return new WaitForSeconds(2);

    for (int i = 0; i <= MAX_FRAG_CNT; ++i)
    {
        posList.Clear();
        for (int j = i; j <= MAX_FRAG_CNT; ++j)
        {
            posList.Add(cubicBezier(fromPos, handPos1, handPos2, toPos, (float)j / MAX_FRAG_CNT));
        }
        lineRenderer.positionCount = posList.Count;
        lineRenderer.SetPositions(posList.ToArray());
        yield return new WaitForSeconds(0.001f);
    }
    Destroy(gameObject);
}

改下Earth脚本中的调用,如下

// Earth.cs

/// <summary>
/// 发射辐射射线
/// </summary>
/// <returns></returns>
IEnumerator FireLine()
{
	this.line.gameObject.SetActive(false);
	while (true)
	{
  		...
  		// 启动协程
        StartCoroutine(line.DrawRay(from, ctrlPoint1, ctrlPoint2, to));

        // 随机等待一个事件
        yield return new WaitForSeconds(Random.Range(0.1f, 0.5f));
   }
}

2、运行效果

运行效果如下,可以,
请添加图片描述

五、加点粒子特效

制作个粒子效果,
请添加图片描述
包装成预设,挂到Line节点下,作为起始点和目标点的特效,如下
在这里插入图片描述
Line脚本中加上粒子控制的逻辑,

// Line.cs

public Transform startPoint;
public Transform endPoint;

/// <summary>
/// 绘制曲线
/// </summary>
/// <param name="fromPos">起始坐标</param>
/// <param name="ctrlPoint1">控制点1</param>
/// <param name="ctrlPoint2">控制点2</param>
/// <param name="toPos">目标坐标</param>
public IEnumerator DrawRay(Vector3 earthPos, Vector3 fromPos, Vector3 handPos1, Vector3 handPos2, Vector3 toPos)
{
    // 起始位置粒子
    startPoint.gameObject.SetActive(true);
    startPoint.forward = fromPos - earthPos;
    startPoint.localPosition = fromPos + startPoint.forward * 0.01f;
    for (int i = 0; i <= MAX_FRAG_CNT; ++i)
    {
        posList.Clear();
        for (int j = 0; j <= i; ++j)
        {
            posList.Add(cubicBezier(fromPos, handPos1, handPos2, toPos, (float)j / MAX_FRAG_CNT));
        }
        lineRenderer.positionCount = posList.Count;
        lineRenderer.SetPositions(posList.ToArray());
        yield return new WaitForSeconds(0.02f);
    }

    // 目标位置粒子
    endPoint.gameObject.SetActive(true);
    endPoint.forward = toPos - earthPos;
    endPoint.localPosition = toPos + endPoint.forward * 0.01f;
    yield return new WaitForSeconds(2);
    startPoint.gameObject.SetActive(false);
    for (int i = 0; i <= MAX_FRAG_CNT; ++i)
    {
        posList.Clear();
        for (int j = i; j <= MAX_FRAG_CNT; ++j)
        {
            posList.Add(cubicBezier(fromPos, handPos1, handPos2, toPos, (float)j / MAX_FRAG_CNT));
        }
        lineRenderer.positionCount = posList.Count;
        lineRenderer.SetPositions(posList.ToArray());
        yield return new WaitForSeconds(0.001f);
    }
    Destroy(gameObject);
}

最终运行效果如下
请添加图片描述

六、工程源码

本文工程我已上传到GitCode,感兴趣的同学可自行下载学习,
地址:https://gitcode.net/linxinfa/UnityEarthRay
注:我使用的Unity版本是2021.1.7.f1c1
在这里插入图片描述

七、完毕

好了,就写到这里吧。
我是新发,https://blog.csdn.net/linxinfa
一个在小公司默默奋斗的Unity开发者,希望可以帮助更多想学Unity的人,共勉~

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

【游戏开发实战】Unity实现类似GitHub地球射线的效果(LineRenderer | 贝塞尔曲线) 的相关文章

  • [大模拟]Test Week14 猫睡觉问题

    目录 大模拟 猫睡觉问题题意样例样例输入 xff1a 样例输出 思路总结代码 大模拟 猫睡觉问题 题意 众所周知 xff0c TT家里有一只魔法喵 这只喵十分嗜睡 一睡就没有白天黑夜 喵喵一天可以睡多次 xff01 xff01 每次想睡多久
  • [区间DP/状压DP]Exercise Week12 A~E

    目录 A 水 找数题意样例样例输入 xff1a 样例输出 思路总结代码 B BFS 逃离题意样例样例输入 xff1a 样例输出 xff1a 思路总结代码 C DP 扫楼题意样例样例输入 xff1a 样例输出 思路总结代码 D 区间DP 最长
  • 新的开始( [USACO08OCT]打井Watering Hole)

    新的开始 newstart pas c cpp 题目描述 话说小 FF 在经历了上次 寻找古代王族遗产 的探险后 xff0c 成为了世界上最伟大的探险 家并拥有了一大笔财富 当然他不能坐吃山空 xff0c 必须创造财富 xff01 xff0
  • UOS配置本地APT源和外部软件包

    root 64 skill PC mount dev sr0 mnt mount mnt WARNING device write protected mounted read only root 64 skill PC vi etc ap
  • [二分答案] 洛谷P1873 砍树

    目录 题意样例样例输入 xff1a 样例输出 思路总结代码 题意 伐木工人米尔科需要砍倒M米长的木材 这是一个对米尔科来说很容易的工作 xff0c 因为他有一个漂亮的新伐木机 xff0c 可以像野火一样砍倒森林 不过 xff0c 米尔科只被
  • [区间DP]洛谷P1063 能量项链

    目录 题意样例样例输入 xff1a 样例输出 思路总结代码 题意 样例 样例输入 xff1a 4 2 3 5 10 样例输出 710 思路 1 经典区间DP题 算是合并石子的变种 只不过由一个点变成了一个区间 不过我们也可以用结构体存储 当
  • Linux命令行初接触-1 操作文件和目录

    操作文件 amp 目录 1 通配符含义常用通配符常用字符类类型匹配范例 2 mkdir 创建目录3 cp 复制文件和目录工作方式常用选项 4 mv 移动和重命名文件工作方式常用选项 5 rm 删除文件和目录工作方式常用选项注意事项 6 ln
  • 机器学习入入入入门(1)机器学习基本概念、引出深度学习

    机器学习入入入入门 xff08 1 xff09 0 前言1 基本步骤2 基本概念2 1 Hyperparameters2 2 local minima 3 linear model3 1 基础概念 4 piecewise linear cu
  • 深度学习蒟蒻入门——从0安装pytorch(CPU版)

    从0安装pytorch 1 检查自己的电脑有没有GPU2 安装CPU版的pytorch3 测试pytorch 1 检查自己的电脑有没有GPU 首先打开任务管理器 xff0c 选择性能栏 然后滑到最下 xff0c 看是否有GPU一项 xff0
  • 系统学习iOS动画 —— Stroke和路径动画

    这是要完成的动画 xff1a 先添加需要的代码 xff0c 这里需要将storyboard的ViewController换成TableViewController xff0c 将Under Top Bars 和 Under Bottom B
  • 不知道这些网站还做什么程序员呀!

    今天我就来总结一些程序员必备的网站 xff0c 囊括开源项目 解决bug 技术分享 一线资源和自我提升的网站 xff0c 希望能对广大程序猿有所帮助 xff0c 赶紧给我收藏起来 xff0c 下次刷不到了可别说我没提醒你 我们首先来看一下国
  • (音视频开发)WebRTC进阶流媒体服务器开发-多人互动架构

    一 xff1a 多人互动架构方案 xff08 一 xff09 WebRTC回顾 xff0c 两层含义 xff1a 1 WebRTC是google开源的流媒体客户端 xff0c 可以进行实时通讯 xff0c 主要应用于浏览器之间进行实时通讯
  • 10种linux下磁盘快照方式恢复系统

    导读大家都知道windows系统有一个磁盘快照的功能 xff0c 在windows2003中系统恢复开始依赖于一个叫做硬盘快照服务 Volume Snapshot Service 的服务 xff0c 他能够自动创建系统快照 包括正在使用的文
  • ubuntu安装go开发环境

    一 为ubuntu20 04更新源 给root用户设置密码 xff1a 命令 xff1a sudo passwd root 备份原来的源 xff0c 命令 xff1a sudo cp etc apt sources list etc apt
  • 如何修复Ubuntu中包缓存文件被毁问题

    导读今天 xff0c 我尝试更新我的 Ubuntu 18 04 LTS 的仓库列表 xff0c 但收到了一条错误消息 xff1a E The package cache file is corrupted it has the wrong
  • 1002 A+B for Polynomials (25分) 详解+易错点

    注意点 xff1a 系数为0 xff0c 则不输出 xff0c 例 xff1a 其中 1和1相加为0 xff0c 则在输出时避免这一项 xff0c 而且要注意结果的K值 xff0c 不要包括这一项 xff0c 思路 xff0c 利用结构体存
  • Linux远程桌面的选择

    Linux的远程桌面主要分两个部分 xff1a Linux客户机连Linux服务器和Windows客户机连Linux服务器 xff0c 还有现在用平板电脑连远程桌面 Linux客户机连Windows服务器比较简单没啥可说的 xff0c rd
  • Kali Linux mdk3WiFi洪水攻击 攻击路由器 生成虚假WiFi WiFi身份验证攻击可使连接WiFi的手机掉线重连抓包

    将无线网卡转换为监听模式 airmon ng start wlan0 查找附近无线网络 airodump ng wlan0mon Authentication DoS xff1a xff08 洪水攻击 xff0c 又叫做身份验证攻击 xff
  • 大一java程序设计的某次作业题解

    题目描述 xff1a 设计程序实现输入日期及机票张数 xff0c 计算出应付金额 假设北京至上海的机票全价为 1200 元 张 xff0c 以 2017 年为例进行程序编写 xff0c 所有的法定假日 xff0c 机票无折扣 xff1b 除
  • D. Make It Round(1759D)

    要求n k后缀0数量最多 xff08 k lt 61 m xff09 xff0c 且n k尽可能大 比赛时思路 xff08 错误 xff09 xff1a 10是由2和5组成 xff0c 故先统计n的因子包含2的个数num2 包含5的个数nu

随机推荐