Unity 动态网格形变效果

 

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

public class MeshChangeDemo : MonoBehaviour
{
    public MeshFilter meshFilter;
    public float changeForce = 10f;
    public float springForce = 20f;
    public float forceOffset = 0.1f;
    public float damping = 5f;
    public float uniformScale = 1f;
    public float uniformScaleRate = 0f;
    private Mesh mesh;
    private Vector3[] originVertices;
    private Vector3[] displayVertices;
    private Vector3[] vertexVelocities;
    // Start is called before the first frame update
    void Start()
    {
        uniformScale = transform.localScale.x;
        mesh = meshFilter.mesh;
        originVertices = (Vector3[])mesh.vertices.Clone();
        displayVertices = (Vector3[])mesh.vertices.Clone();
        vertexVelocities = new Vector3[displayVertices.Length];
    }

    // Update is called once per frame
    void Update()
    {
        Ray ray = Camera.main.ScreenPointToRay(Input.mousePosition);
        RaycastHit hit;
        if(Physics.Raycast(ray, out hit))
        {
            MeshChangeDemo meshChangeDemo = hit.transform.GetComponent<MeshChangeDemo>();
            if (meshChangeDemo != null)
            {
                meshChangeDemo.ChangeMesh(hit.point + hit.normal * forceOffset, changeForce);
            }
        }

        for (int i = 0; i < displayVertices.Length; i++)
        {
            Vector3 v = vertexVelocities[i];
            Vector3 displacement = displayVertices[i] - originVertices[i];
            v -= displacement * springForce * Time.deltaTime; //回弹形状
            v *= 1f - damping * Time.deltaTime;               //阻尼
            vertexVelocities[i] = v;
            //displayVertices[i] += v * (Time.deltaTime / uniformScale); //应用速度 除以uniformScale是为了兼容缩放了物体的情况 可理解为放大物体则形变更小,反之更大
            float uniform = (uniformScale * uniformScaleRate);
            if (uniform == 0)
            {
                uniform = 1;
            }
            displayVertices[i] += v * (Time.deltaTime / uniform);
        }
        mesh.vertices = displayVertices;
        mesh.RecalculateNormals();
    }

    public void ChangeMesh(Vector3 point, float force)
    {
        point = transform.InverseTransformPoint(point);
        for (int i = 0; i < displayVertices.Length; i++)
        {
            Vector3 pointToVertex = displayVertices[i] - point;
            //注意:物体缩放了,但是这里获取到的物体网格顶点实际是没有缩放时的顶点位置,也就是会是得到的一个错误的向量 如果你的物体被缩放了。
            pointToVertex *= uniformScale;//处理物体被缩放,但是模型本身顶点没缩放 导致向量与实际不符,现在乘上一个缩放系数它自身 如果是2倍则是会将向量还原到正确的倍数            
            float f = force / (1f + pointToVertex.sqrMagnitude); //Fv = F / (1 + d^2) 获取顶点受力公式
            float v = f * Time.deltaTime;                        //dv = a * dt  a = F/m 设m=1,则 dv = F * dt
            vertexVelocities[i] += pointToVertex.normalized * v;
        }
    }

}

从上至下参数含义:施加力、回弹力、施加力相对碰撞点的法线偏移量、阻尼、缩放系数(动态获取物体局部缩放值X)、缩放系数的影响系数。可自行调整达到最优效果。代码参考UnityMesh编程(4)网格变形 - 知乎

 

  • 8
    点赞
  • 39
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Unity是一款强大的游戏开发引擎,它在物理模拟和图形渲染方面具有出色的表现。在Unity中,我们可以通过使用材质、纹理和粒子系统等功能来实现动态水体涟漪效果。 首先,我们需要创建一个水面模型,可以使用Unity内置的平面网格,或者导入自定义的水体模型。然后,我们可以为水面添加一个适用于涟漪效果的材质。这个材质应该包括透明度、反射和折射等属性,以实现真实的水体效果。 为了实现动态涟漪效果,我们可以使用Unity的粒子系统。在水体表面的特定位置,通过发射水滴粒子,我们可以模拟水面上的涟漪效果。这些粒子应该具有适当的速度和角度,以便能够溅起涟漪效果。 此外,我们还可以利用Unity的碰撞检测功能,使得水滴粒子与水体表面发生碰撞时产生涟漪效果。当水滴粒子与水面碰撞时,我们可以通过修改水面的顶点位置,以及更新法线和切线来实现漂移和波纹的效果。这样,当水滴在水面上溅起时,周围的水面将产生波动,并形成涟漪效果。 为了增强涟漪效果的真实感,我们可以通过调整涟漪的扩散范围、持续时间和淡化效果来控制涟漪的外观。另外,还可以使用图形着色器和纹理贴图来增加水体的细节和真实感。 总之,在Unity中实现动态水体涟漪效果,我们可以使用材质、粒子系统、碰撞检测和纹理贴图等功能来模拟真实的涟漪效果。这些功能可以帮助我们创造出逼真的水体效果,提升游戏或仿真场景的视觉体验。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值