【Unity小技巧】unity2d平台制作一根三七调的鱼竿效果(附git源码)

文章目录

  • 前言
  • 素材
  • 开始
  • 源码
  • 参考
  • 完结

前言

今天我们做一个2d鱼竿的效果,先看一下效果成品效果
在这里插入图片描述

素材

鱼竿
在这里插入图片描述

开始

首先创建一个2D URP项目 ,创建一个空物体作为鱼竿,并创建两个子物体作为开始和结束点
配置层级如下
在这里插入图片描述
鱼竿和鱼线加Line Renderer
在这里插入图片描述
新增鱼竿材质
在这里插入图片描述
贝塞尔曲线基类

using System.Collections;
using System.Collections.Generic;
using UnityEngine;/// <summary>
/// 贝塞尔曲线 动态.
/// </summary>
public class BezierHelper{// 一阶贝塞尔曲线,参数P0、P1、t对应上方原理内的一阶曲线参数.public static Vector3 Bezier(Vector3 p0, Vector3 p1, float t){return (1 - t) * p0 + t * p1;}// 二阶贝塞尔曲线,参数对应上方原理内的二阶曲线参数.Vector3 Bezier(Vector3 p0, Vector3 p1, Vector3 p2, float t){Vector3 p0p1 = (1 - t) * p0 + t * p1;Vector3 p1p2 = (1 - t) * p1 + t * p2;Vector3 temp = (1 - t) * p0p1 + t * p1p2;return temp;}// 三阶贝塞尔曲线,参数对应上方原理内的三阶曲线参数.public static  Vector3 Bezier(Vector3 p0, Vector3 p1, Vector3 p2, Vector3 p3, float t){Vector3 temp;Vector3 p0p1 = (1 - t) * p0 + t * p1;Vector3 p1p2 = (1 - t) * p1 + t * p2;Vector3 p2p3 = (1 - t) * p2 + t * p3;Vector3 p0p1p2 = (1 - t) * p0p1 + t * p1p2;Vector3 p1p2p3 = (1 - t) * p1p2 + t * p2p3;temp = (1 - t) * p0p1p2 + t * p1p2p3;return temp;}// 多阶贝塞尔曲线,使用递归实现.public static Vector3 Bezier(float t, List<Vector3> p){if (p.Count < 2)return p[0];List<Vector3> newp = new List<Vector3>();for (int i = 0; i < p.Count - 1; i++){Debug.DrawLine(p[i], p[i + 1]);Vector3 p0p1 = (1 - t) * p[i] + t * p[i + 1];newp.Add(p0p1);}return Bezier(t, newp);}
}

新建FishingRodCtrl脚本,使用贝塞尔曲线控制鱼竿弯曲

using System.Collections;
using System.Collections.Generic;
using System.Security.Cryptography.X509Certificates;
using UnityEngine;
using System.Linq;
using UnityEngine.UIElements;[System.Serializable] // 序列化类,使其在Unity编辑器中可见
public class CtrlPointInfo{ // 控制点信息类public float DiffH; // 水平偏移量public float DiffV; // 垂直偏移量}
public class FishingRodCtrl : MonoBehaviour
{private LineRenderer mLr; // 线渲染器private LineRenderer mLr_line;//鱼线的线渲染器private Transform mBeginTran; // 钓鱼竿开始的位置private Transform mEndTran; // 钓鱼竿结束的位置private float mStrength; // 钓鱼竿的受力大小private Transform mTarget; // 钓鱼竿的目标位置public List<CtrlPointInfo> CtrlPointInfos;//控制点的偏移信息列表void Start(){mLr = GetComponent<LineRenderer>();; // 获取线渲染器组件mBeginTran = transform.GetChild(0); // 获取第一个子物体的Transform组件,即钓鱼竿开始的位置mEndTran = transform.GetChild(1); // 获取第二个子物体的Transform组件,即钓鱼竿结束的位置mTarget = GameObject.Find("Target").GetComponent<Transform>(); // 找到名为"Target"的游戏对象,并获取其Transform组件,即钓鱼竿的目标位置mLr_line = GameObject.Find("鱼线").GetComponent<LineRenderer>(); // 找到名为"鱼线"的游戏对象,并获取其线渲染器组件}void Update(){DrawFishingRod();DrawFishingLine();}//绘制鱼竿private void DrawFishingRod(){var drawPoints = GetDarwFishingRodPoints(); // 获取绘制钓鱼竿的点mLr.positionCount = drawPoints.Count; // 设置线渲染器的顶点数mLr.SetPositions(drawPoints.ToArray()); // 设置线渲染器的所有顶点的位置}//绘制鱼线private void DrawFishingLine(){Vector3[] points = new Vector3[2]{mEndTran.position, mTarget.position}; // 创建一个包含钓鱼竿结束的位置和目标位置的向量数组mLr_line.positionCount = 2; // 设置鱼线的线渲染器的顶点数为2mLr_line.SetPositions(points); // 设置鱼线的线渲染器的所有顶点的位置}//获取绘制鱼竿的点private List<Vector3> GetDarwFishingRodPoints(){mStrength = Mathf.Clamp(Vector2.Distance(mEndTran.position, mTarget.position) - 5, 0, 100) * 0.2f; // 计算力的大小,通过目标位置和钓鱼竿结束的位置的距离,然后减去5,最后乘以0.2List<Vector3> points = new List<Vector3>(); // 创建一个新的向量列表,用于存储点//获取控制点List<Vector3> mCtrlPoints = new List<Vector3>();mCtrlPoints.Add(mBeginTran.position); // 将钓鱼竿开始的位置添加到控制点列表中//循环创建控制点for (int i = 0; i < CtrlPointInfos.Count; i++) // 遍历控制点的偏移信息列表{Vector3 ctrlPoint = mBeginTran.position + (mEndTran.position - mBeginTran.position) * CtrlPointInfos[i].DiffH; // 计算控制点的位置,首先进行水平方向的偏移ctrlPoint = ctrlPoint + (Vector3)Vector2.Perpendicular(mEndTran.position - mBeginTran.position) * CtrlPointInfos[i].DiffV * -1 * mStrength; // 然后进行垂直方向的偏移mCtrlPoints.Add(ctrlPoint); // 将计算出的控制点添加到控制点列表中}//最好一个控制点单独控制,从鱼竿竿稍 向力的反方向偏移,可以使曲线顺滑Vector3 lastCtrlPoint = mEndTran.position + (mEndTran.position - mTarget.position).normalized * CtrlPointInfos.Last().DiffV * mStrength;mCtrlPoints.Add(mEndTran.position); // 将钓鱼竿结束的位置添加到控制点列表中for (int i = 0; i < 100; i++){ // 遍历100次points.Add(BezierHelper.Bezier(i/100f, mCtrlPoints)); // 使用贝塞尔曲线算法计算点的位置,并添加到点列表中}return points; // 返回点列表}
}

挂载脚本,配置参数
在这里插入图片描述
运行效果

在这里插入图片描述

源码

https://gitcode.net/unity1/unity2d-fishing
在这里插入图片描述

参考

【视频】https://www.bilibili.com/video/BV1QL411q7Tr

完结

赠人玫瑰,手有余香!如果文章内容对你有所帮助,请不要吝啬你的点赞评论和关注,以便我第一时间收到反馈,你的每一次支持都是我不断创作的最大动力。当然如果你发现了文章中存在错误或者有更好的解决方法,也欢迎评论私信告诉我哦!

好了,我是向宇,https://xiangyu.blog.csdn.net

一位在小公司默默奋斗的开发者,出于兴趣爱好,于是最近才开始自习unity。如果你遇到任何问题,也欢迎你评论私信找我, 虽然有些问题我可能也不一定会,但是我会查阅各方资料,争取给出最好的建议,希望可以帮助更多想学编程的人,共勉~
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/54097.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

windows一键启动jupyter

windows一键启动jupyter jupyter简介 Jupyter是一个开源的交互式计算环境&#xff0c;主要用于数据分析、数据可视化和科学计算。它的名字来源于三种编程语言的缩写&#xff1a;Julia、Python和R&#xff0c;这三种语言都可以在Jupyter环境中运行。如果您想进行数据分析、科学…

为什么网络互联地址设置为30位地址

对于点对点链路&#xff0c;为了节约IPv4地址&#xff0c;一般为其分配/30地址块&#xff0c;这样包含4个地址&#xff1a;最小地址作为网络地址&#xff0c;最大地址作为广播地址&#xff0c;剩余两个可分配地址&#xff0c;分配给链路两端的接口&#xff0c;这是最普遍的方法…

大语言模型之六- LLM之企业私有化部署

数据安全是每个公司不得不慎重对待的&#xff0c;为了提高生产力&#xff0c;降本增效又不得不接受新技术带来的工具&#xff0c;私有化部署对于公司还是非常有吸引力的。大语言模型这一工具结合公司的数据可以大大提高公司生产率。 私有化LLM需要处理的问题 企业内私有化LLM…

【Axure原型分享】能统计中英文字数的多行输入框

今天和大家分享能统计中英文字数的多行输入框的原型模板&#xff0c;在输入框里输入内容后&#xff0c;能够动态根据输入框的内容&#xff0c;统计出字符数量&#xff0c;包括总字数、中文字数、英文字数、数字字数、其他标点符号的字数&#xff0c;具体效果可以观看下方视频或…

系统架构设计高级技能 · 安全架构设计理论与实践

系列文章目录 系统架构设计高级技能 软件架构概念、架构风格、ABSD、架构复用、DSSA&#xff08;一&#xff09;【系统架构设计师】 系统架构设计高级技能 系统质量属性与架构评估&#xff08;二&#xff09;【系统架构设计师】 系统架构设计高级技能 软件可靠性分析与设计…

深入理解高并发编程 - Thread 类的 stop () 和 interrupt ()

stop() stop() 方法被用于停止线程。然而&#xff0c;需要注意的是&#xff0c;stop() 方法已经被标记为已废弃&#xff08;deprecated&#xff09;&#xff0c;并且不推荐使用。这是因为使用该方法可能导致不可预料的问题和数据不一致性&#xff0c;因此它被认为是不安全的。…

Spring MVC 二 :基于xml配置

创建一个基于xml配置的Spring MVC项目。 Idea创建新项目&#xff0c;pom文件引入依赖&#xff1a; <dependency><groupId>org.springframework</groupId><artifactId>spring-context</artifactId><version>5.2.12.RELEASE</version>…

【C++】AVL树(高度平衡二叉树)

AVL树 概念AVL树节点定义AVL树节点插入AVL树四种旋转情况左单旋右单旋先左单旋再右单旋先右单旋后左单旋 元素的插入及控制平衡判断最后节点是否平衡 概念 二叉搜索树虽然可以缩短查找的效率&#xff0c;但如果数据有序或者接近有序二叉搜索树将退化为单支树&#xff0c;查找元…

R语言中<- 的含义

一般语言的赋值是 号&#xff0c;但是 R 语言是数学语言&#xff0c;所以赋值符号与我们数学书上的伪代码很相似&#xff0c;是一个左箭头 <- &#xff1a; 举个例子&#xff1a; a <- 12 b <- 45 print(a b) 以上代码执行结果&#xff1a;57 这个赋值符号是 R …

copy is all you need前向绘图 和疑惑标记

疑惑的起因 简化前向图 GPT4解释 这段代码实现了一个神经网络模型&#xff0c;包含了BERT、GPT-2和MLP等模块。主要功能是给定一个文本序列和一个查询序列&#xff0c;预测查询序列中的起始和结束位置&#xff0c;使其对应文本序列中的一个短语。具体实现细节如下&#xff1a…

Win系统设置开机自启项及自定义自启程序

Win系统设置开机自启项及自定义自启程序 分用户自启动和系统自启动两种形式&#xff1a; 1. 用户自启动目录&#xff1a;C:\Users\Administrator\AppData\Roaming\Microsoft\Windows\Start Menu\Programs\Startup 用快速键打开&#xff1a; Win键R键&#xff0c;输入shell:…

JavaScript 基础知识回顾与复习---闭包

当我们说到闭包&#xff0c;在JavaScript中闭包是一个让人难以理解甚至说是一个近乎神话的概念。闭包往往也是面试必考的题目&#xff0c;如果能够掌握闭包对我们自己来说那也是一种极大的提升。在学习的过程中不要害怕闭包&#xff0c;闭包并不是一个新的语法或者模式&#xf…

sql server 快速安装

目录标题 一、下载二、直接选择基本安装二、下载ssms&#xff08;数据库图形化操作页面&#xff09;三、开启sa账号认证&#xff08;一&#xff09;第一步&#xff1a;更改身份验证模式&#xff08;二&#xff09;第二步&#xff1a;启用 sa 登录四、开启tcp/ip 一、下载 下载…

低通滤波器和高通滤波器

应用于图像低通滤波器和高通滤波器的实现 需要用到傅里叶变换 #include <opencv2/opencv.hpp> #include <Eigen> #include <iostream> #include <vector> #include <cmath> #include <complex>#define M_PI 3.14159265358979323846…

QT5.12.12通过ODBC连接到GBase 8s数据库(CentOS)

本示例使用的环境如下&#xff1a; 硬件平台&#xff1a;x86_64&#xff08;amd64&#xff09;操作系统&#xff1a;CentOS 7.8 2003数据库版本&#xff08;含CSDK&#xff09;&#xff1a;GBase 8s V8.8 3.0.0_1 为什么使用QT 5.12.10&#xff1f;该版本包含QODBC。 1&#…

ES6中promise的使用

ES6中promise的使用 本文目录 ES6中promise的使用基础介绍箭头函数function函数状态 原型方法Promise.prototype.then()Promise.prototype.catch() 静态方法Promise.all()Promise.race()Promise.any() 链式回调 基础介绍 官网&#xff1a;https://promisesaplus.com/ window.…

最新docker多系统安装技术

在Ubuntu操作系统中安装Docker 在Ubuntu操作系统中安装Docker的步骤如下。 1&#xff0e;卸载旧版本Docker 卸载旧版本Docker的命令如下&#xff1a; $ sudo apt-get remove docker docker-engine docker.io 2&#xff0e;使用脚本自动安装 在测试或开发环境中&#xff0…

STM32 进不了main 函数

1. 我用的是STM32L151C8T6 的芯片&#xff0c;在github 上找了个别人的例程&#xff0c;拿来当模板改&#xff0c;由于他用的是HSE 外部晶振&#xff0c;我用的是内部晶振HSI&#xff0c;所以需要改系统时钟&#xff0c;改完后debug&#xff0c; 一直进不了main 函数&#xff0…

c语言练习题29:获得月份天数

获得月份天数 代码&#xff1a; //法一 #include<stdio.h> int main() {int y 0;int m 0;int days[13] { 0,31,28,31,30,31,30,31,31,30,31,30,31 };while (~scanf("%d%*c%d", &y, &m)) {int day days[m];if ((y % 4 0) && ((y % 400 …

图像线段检测几种方法

1、方法一 当我将OpenCV提升到4.1.0时&#xff0c;LineSegmentDetector&#xff08;LSD&#xff09;消失了。 OpenCV-contrib有一个名为FastLineDetector的东西&#xff0c;如果它被用作LSD的替代品似乎很好。如果你有点感动&#xff0c;你会得到与LSD几乎相同的结果。 2、方…