Unity 软性管的实现

news2025/1/11 20:45:24

概述

因近期项目有要求使用到水管这种软性管的模拟,该篇主要说明软管的实现和应用,参考自:unity3D---实现柔软水管(蛇的移动)效果一(无重力)_unity 软管_ayouayouwei的博客-CSDN博客

效果

 实现代码

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

public class HoseScript : MonoBehaviour
{
    [Header("起点位置")]
    public Vector3 StartPos = new Vector3(0, 0, 0);

    [Header("目标物体-终点物体")]
    public GameObject role;
    public List<Segment2> segments;

    [Header("长度(0-1)")]
    public float Length;
    [Header("宽度(0-1)")]
    public Vector3 Size;
    [Header("曲率(20-100)")]
    public float Step;


    void Start()
    {

        segments = new List<Segment2>();

        for (int i = 0; i < Step; i++)
        {
            if (i == 0)
            {
                var seg = new Segment2(new Vector3(0, 1, 0.5f), Length, 0, 0, transform, Size);
                segments.Add(seg);
                seg.Set();
            }
            else
            {
                Segment2 seg = new Segment2(segments[i - 1], Length, 0, 0, transform, Size);
                segments.Add(seg);
                seg.Set();
            }
        }
    }

    private void Update()
    {
        OnPeopleRun();
    }

    public void OnPeopleRun()
    {
        for (int i = segments.Count - 1; i >= 0; i--)
        {
            if (i == segments.Count - 1)
            {
                segments[i].Follow(role.transform.position);
            }
            else
            {
                segments[i].Follow(segments[i + 1].a);
            }
            segments[i].update();
        }

        segments[0].SetA(StartPos);


        for (int i = 1; i < segments.Count; i++)
        {
            segments[i].SetA(segments[i - 1].b);

        }
    }
}

//用于处理每一段
public class Segment2
{
    public Vector3 a;         //起点
    public Vector3 b;         //终点
    public float len;           //长度
    public float angle1;        //向量b-a  在xoz面的投影,与x轴的夹角
    public float angle2;        //向量b-a 与y轴的夹角
    public GameObject sphere;   //关节处添加一个球体
    public GameObject sylinder = null;   //每一段用一个圆柱体来连接
    public Transform _par;
    public Vector3 size;

    public Segment2(Segment2 parent_, float len_, float angle1_, float angle2_, Transform par,Vector3 _size)
    {
        a = new Vector3(parent_.b.x, parent_.b.y, parent_.b.z);
        len = len_;
        angle1 = angle1_;
        angle2 = angle2_;
        _par = par;
        size = _size;
        calculateB();
        sphere = GameObject.CreatePrimitive(PrimitiveType.Sphere);
        sphere.transform.parent = _par;
        sphere.transform.position = a;
        sphere.transform.localScale = _size;
        sphere.GetComponent<SphereCollider>().enabled = false;
        DrawLine();
    }
    public Segment2(Vector3 vector3, float len_, float angle1_, float angle2_, Transform par, Vector3 _size)
    {
        a = vector3;
        len = len_;
        angle1 = angle1_;
        angle2 = angle2_;
        _par = par;
        size = _size;
        calculateB();
        sphere = GameObject.CreatePrimitive(PrimitiveType.Sphere);
        sphere.transform.parent = _par;
        sphere.transform.localPosition = a;
        sphere.transform.localScale = _size;
        sphere.GetComponent<SphereCollider>().enabled = false;
        DrawLine();
    }
    public void calculateB()
    {
        float dy = len * Mathf.Cos(angle2);
        float dx = len * Mathf.Sin(angle2) * Mathf.Cos(angle1);
        float dz = len * Mathf.Sin(angle2) * Mathf.Sin(angle1);
        b = new Vector3(a.x + dx, a.y + dy, a.z + dz);

    }
    //目标位置pos,根据目标位置确定a的位置
    public void Follow(Vector3 pos)
    {
        Vector3 dir = pos - a; //单位向量

        angle1 = Mathf.Atan2(dir.z, dir.x);
        angle2 = Mathf.Atan2(Mathf.Sqrt(dir.x * dir.x + dir.z * dir.z), dir.y);
        dir = Vector3.Normalize(dir);
        dir = dir * len * (-1f);
        a = pos + new Vector3(dir.x, dir.y, dir.z);

    }

    public void SetA(Vector3 a_)
    {
        a = a_;
        update();
    }
    public void update()
    {
        calculateB();
        sphere.transform.position = a;
        DrawLine();
    }

    //根据起点和终点来确定圆柱体的位置和旋转角度
    public void DrawLine()
    {
        if (sylinder == null)
        {
            sylinder = GameObject.CreatePrimitive(PrimitiveType.Cylinder);
            sylinder.transform.parent = _par;
        }
        Vector3 position = (b + a) / 2.0f;
        Vector3 rotation = b - a;
        float length = Vector3.Distance(a, b) / 2;
        sylinder.transform.position = position;
        sylinder.transform.localRotation = Quaternion.FromToRotation(Vector3.up, rotation);
        //sylinder.transform.localScale = new Vector3(1, length, 1);

    }

    public void Set()
    {
        sylinder.transform.localScale = size;
        sylinder.GetComponent<CapsuleCollider>().enabled = false;
    }

}

1.在Unity中创建一个空物体,挂载HoseScript.cs脚本

2.创建一个基础物体,将基础物体赋值到HoseScript.cs的Role参数上

 运行后在Scene视图下拖拽Role即可看到实现效果,具体实现思路可参考文中第一段的原创链接,本篇只是根据原创进行二次整合后的使用层讲解。

整合了Demo样例只要一点点积分,如果没有积分也可私聊我免费获取

Unity软性管的实现资源-CSDN文库

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

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

相关文章

B/S医院手术麻醉管理系统源码:麻醉知情同意书模板

麻醉知情同意书模板 姓名:​ 性别:​ 年龄:​ 科别:​ 床号:​ 住院号:​ 疾病介绍和治疗建议: 医生已告知我因​手术&#xff0c;而接受麻醉。 1.麻醉作用的产生主要是利用麻醉药使中枢神经系统或神经中某些部位受到抑制的结果&#xff0c;临床麻醉的主要任务是: 2.为…

webpack 的打包流程

1.webpack 的打包流程 从以上5个方面来分析Webpack的打包流程&#xff1a; 初始化参数&#xff1a;这一步会从我们配置的webpack.config.js中读取到对应的配置参数和shell命令中传入的参数进行合并得到最终打包配置参数。 开始编译&#xff1a;这一步我们会通过调用webpack()方…

计算机网络基础知识(二)—— 什么是Ip地址、Mac地址、网关、子网掩码、DNS

文章目录 01 | Ip地址02 | Mac地址03 | 网关04 | 子网掩码05 | DNS06 | 总结 初次接触网络时&#xff0c;只知道电脑连接网线&#xff0c;就可以打开4399玩小游戏&#xff0c;可以登录QQ和朋友聊天&#xff1b; 再次接触网络时&#xff0c;知道了怎么查看自己电脑的网络情况&am…

06 - 5 生产者消费者模式

架构演进 介绍 同步调用变成异步调用生产数据与消费数据分离协调不同处理速度 生产者 系统运转的动力为下一个环节产生待处理的工作/数据与消费者的关系 重点在如何将数据发送到容器对消费者无依赖不关注消费者的how/when 发送顺序 消费者 容器 平衡 与EDA对比 消费策略 优点…

我发布了自己第一个由ChatGPT辅助开发的开源项目goattribute

需求产生 前两天在工作过程中又遇到了一直以来困惑我的一个问题&#xff0c;就是Go配置项的管理问题。 在开发一个新项目的时候&#xff0c;往往涉及到配置项的管理。个人小项目可能会通过配置文件来传入、环境变量来传入&#xff0c;也可能通过命令行参数来传入&#xff0c;公…

代码随想录 LeetCode数组篇 长度最小的子数组

文章目录 &#xff08;中等&#xff09;209. 长度最小的子数组&#xff08;中等&#xff09;904. 水果成篮&#xff08;困难&#xff09;76. 最小夫覆盖子串 &#xff08;中等&#xff09;209. 长度最小的子数组 我的思路&#xff1a;双指针p和q&#xff0c;滑动窗口的思想 每…

牛客练习赛111 D青蛙兔子的约会

题目链接 示例1 输入 3 3 4 10 1 2 2 4 5 1 1 3 5 11 1 1 输出 YES NO NO 说明 第一问&#xff0c;青蛙晚上向右跳1次&#xff0c;白天无法与兔子相遇。青蛙向右跳2次&#xff0c;也就是2a6的距离&#xff0c;白天兔子向左跳1次&#xff0c;可以相遇。所以在跳[1,2]次中&#…

app持续交付实战

app持续交付实战 一、学习目标二、优势三、子任务拆分四、环境依赖1、安卓 SDK2、安卓设备&#xff08;真机 or 模拟器&#xff09;3、Appium 自动化测试4、JDK5、Python3环境6、allure-commandline工具7、allure插件 五、实战任务&#xff1a;串行执行 Jenkins Pipeline 项目1…

Python学习之批量转换图片格式和统一图片尺寸

前言 大家在工作的时候是不是都会接触到很多的图片&#xff0c;为了满足不同的需求&#xff1a; 兼容性&#xff1a;不同设备和应用程序可能支持不同的图片格式。通过转换图片格式&#xff0c;可以确保在各种设备和应用程序中都能够正确地显示图片。 文件大小&#xff1a;不…

基于matlab 从接收脉冲中提取波形参数

一、前言 现代飞机通常随身携带雷达警告接收器 &#xff08;RWR&#xff09;。RWR检测到雷达发射&#xff0c;并在雷达信号照射到飞机上时警告飞行员。RWR不仅可以检测雷达发射&#xff0c;还可以分析截获的信号并编目信号来自哪种雷达。此示例显示了 RWR 如何估计截获脉冲的参…

10倍速度开发贪吃蛇游戏之AI辅助

今天就来聊聊AI代码辅助神器&#xff0c;即便是零基础也能上手&#xff0c;因为实在是太强了&#xff0c;这以后叫程序员们怎么活啊&#xff01;话不多说&#xff0c;直接上神器 我用的是cursor,其实目前AI辅助代码不止cursor&#xff0c;还有微软家的copilot X这个根植于gith…

JVM-01-JVM知识

1-JVM内存模型 Java开发人员一般情况下&#xff0c;使用过程中&#xff0c;不用关注内存的申请和释放&#xff0c;得益于JVM自动内存分配机制&#xff0c;但是其实是个双刃剑&#xff0c;这可以提升Java开发的效率&#xff0c;但是弱化了开发人员内存管理意识&#xff0c;系统容…

四元数快速入门【Quaternion】

四元数&#xff08;Quaternion&#xff09;是用于旋转和拉伸向量的数学运算符。 本文提供了一个概述&#xff0c;以帮助理解在空间导航等应用程序中对四元数的需求。 推荐&#xff1a;用 NSDT场景设计器 快速搭建3D场景。 可以通过多种方式在空间中准确定位、移动和旋转物体。 …

U盘在电脑上读不出来怎么办?详细解决方法在这!

案例&#xff1a;u盘在电脑上读不出来 【不知道为什么&#xff0c;我把u盘插入电脑后电脑完全读不出。我也不知道到底哪里出现了问题&#xff0c;有人可以帮我解答一下吗&#xff1f;】 U盘作为我们生活中经常使用的存储工具&#xff0c;其能帮我们存储大量的文件&#xff0c…

BM54-三数之和

题目 给出一个有n个元素的数组S&#xff0c;S中是否有元素a,b,c满足abc0&#xff1f;找出数组S中所有满足条件的三元组。 数据范围&#xff1a;0≤n≤1000&#xff0c;数组中各个元素值满足 ∣val∣≤100。 空间复杂度&#xff1a;O(n^2)&#xff0c;时间复杂度 O(n^2)。 注…

React 第三方插件 —— Cron 表达式生成器(qnn-react-cron)

qnn-react-cron 可以看做 react-cron-antd 的升级版&#xff08;具体“渊源”可见文档&#xff09;&#xff0c;现有功能如下&#xff1a; &#x1f389; 全面支持 cron&#xff1a;秒、分、时、日、月、周、年 &#x1f389; 日及周条件互斥&#xff0c;自动改变响应值 &…

8年测试开发,写给1-3年功能测试的几点建议,满满硬货指导

从15年毕业到现在也从业八年了&#xff0c;普通本科毕业&#xff0c;现在一家互联网公司担任测试部门总监&#xff0c;摸爬打滚&#xff0c;坑坑洼洼也经历了不少。思绪很久决定还是写下这篇&#xff0c;希望对后进的小伙子少走一点弯路。 很多人把职场想得太美好&#xff0c;其…

学node写接口!!!

fs 可以读取文档 fs.readFild() 用于读取文件 第一个参数 路径 第二个参数 "utf8"(默认值) 第三个参数 函数 function(err , dataStr ){ 第一个参数是错误&#xff0c; 第二个参数是正确的可以拿到读取文件里面的值 } fs.writeFile() 用于创建文件添加内容 …

【LeetCode中等】1419.数青蛙

给你一个字符串 croakOfFrogs&#xff0c;它表示不同青蛙发出的蛙鸣声&#xff08;字符串 “croak” &#xff09;的组合。由于同一时间可以有多只青蛙呱呱作响&#xff0c;所以 croakOfFrogs 中会混合多个 “croak” 。 请你返回模拟字符串中所有蛙鸣所需不同青蛙的最少数目。…

如何根据参考文献查找原文及详细的文献信息

当我们已知参考文献想要查看下载原文以及相关信息时可以用下面的方法&#xff1a; 例如这篇参考文献&#xff1a;Alsamhi S H, Almalki F, Ma O, et al. Predictive estimation of optimal signal strength from drones over IoT frameworks in smart cities[J]. IEEE Transac…