登录 用户中心() [退出] 后台管理 注册
   
您的位置: 首页 >> SoftHub关联区 >> 主题: [u3d][五星推荐]据说来自 国外某论坛的 平滑法线代码 确实好用     [回主站]     [分站链接]
[u3d][五星推荐]据说来自 国外某论坛的 平滑法线代码 确实好用
clq
浏览(539) - 2020-02-01 21:12:31 发表 编辑

关键字: u3d

[u3d][五星推荐]据说来自 国外某论坛的 平滑法线代码 确实好用

Unity里模型的法线有两种计算方式,第一种是在模型导入选项中进行计算,第二种则是在脚本中调用RecalculateNormal函数进行计算
两种方式虽然都用于计算法线,但使用的算法却大相径庭,下面分别介绍一下
首先说RecalculateNormal函数,调用方式如下所示mesh.RecalculateHardNormals ();
一般在动态创建网格后,会调用这个函数来计算法线,否则整个模型会是同一个颜色,不会有立体效果,此种方式由于是在运行时调用,所以算法比较注重效率
大概就是,对于每个顶点,计算包含它的每个面的法线,求平均
如下:假设有4个顶点 2个面

顶点数组为{0,1,2,3} 这里数字其实是对应的坐标,这里为了方便就用数字代替了
面数组为{0,1,2,0,2,3}其中3个一组,0,1,2为一个面,0,2,3为一个面
声明一个法线数组,长度和顶点数组相同,数组对应同下标的顶点的法线
在计算时,遍历面数组,先计算0,1,2这个面的法线,将其加到法线数组的0,1,2这三个位置
之后计算0,2,3这个面的法线,将其加到法线数组的0,2,3
遍历完成后对法线数组中的每个向量进行归一化,就得到了比较平滑的法线
接下来说模型导入选项中的法线计算
如果用和前面相同的算法,可以发现,我们无法对平滑度进行控制,取极端情况,假如我们要做lowpoly效果,对于这两个面来说,由于共用顶点,所以两个面在0,2这两个位置的颜色始终是一样的,而顶点颜色相同,则意味着两点连线的颜色也是相同的,所以根本无法实现lowpoly效果,如果要实现lowpoly效果,我们必须让两个面在0,2分别有不同的法线,所以我们要加入顶点

现在,面数组变成了{0,1,2,3,4,5}由于没有了共用顶点,所以我们能对每个面的法线进行更好地控制
现在,我们把相同坐标的顶点放在一起,例如下图的ABCD,他们的坐标相同但位于不同面内,

设定一个阈值,比如0
计算cos0 = 1
Foreach m in ABCD
{
Foreach n in ABCD
{
If dot(m法线,n法线)>=1
顶点m对应的法线+=n所在面的法线
}
归一化顶点m对应的法线(此处即求得了最终法线)
}(此处即求得了所有顶点的法线)

先取m为A,然后以n遍历ABCD,计算mn所在的面的法线的点积,点积越小说明两向量夹角越大,点积为1表示向量方向相同,将结果与阈值的cos值进行对比来决定取舍,之后再取B,C,D,分别遍历,最后对所有顶点法线归一化

在阈值为1的情况下,A点的法线即为A所在面的法线,计算结果即为lowpoly效果
下面的代码是国外某论坛的,可以在运行时计算时赋予平滑度
例mesh.RecalculateHardNormals (60);
适用于不同面无共用顶点的情况
对于有共用顶点的情况,只要处理一下就行了

[AppleScript] 纯文本查看 复制代码
for (int i = 0; i < tris.Length; i++) {
vertsLowPoly = verts[tris];
tris[i] = i;
}

注意,代码里申请了数组,这个操作会导致gc,而unity自带的计算方式可能是调用的C++,所以比较高效
[C#] 纯文本查看 复制代码
* The following code was taken from: http://schemingdeveloper.com
*
* Visit our game studio website: http://stopthegnomes.com
*
* License: You may use this code however you see fit, as long as you include this notice
* without any modifications.
*
* You may not publish a paid asset on Unity store if its main function is based on
* the following code, but you may publish a paid asset that uses this code.
*
* If you intend to use this in a Unity store asset or a commercial project, it would
* be appreciated, but not required, if you let me know with a link to the asset. If I
* don't get back to you just go ahead and use it anyway!
*/

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

public static class NormalSolver
{
///
/// Recalculate the normals of a mesh based on an angle threshold. This takes
/// into account distinct vertices that have the same position.
///

///
///
/// The smoothing angle. Note that triangles that already share
/// the same vertex will be smooth regardless of the angle!
///

// Each entry in the dictionary represents a unique vertex position.

mesh.normals = normals;
}
public static void RecalculateNormals(this Mesh mesh, float angle) {
var cosineThreshold = Mathf.Cos(angle * Mathf.Deg2Rad);

var vertices = mesh.vertices;
var normals = new Vector3[vertices.Length];

// Holds the normal of each triangle in each sub mesh.
var triNormals = new Vector3[mesh.subMeshCount][];

var dictionary = new Dictionary>(vertices.Length);

for (var subMeshIndex = 0; subMeshIndex < mesh.subMeshCount; ++subMeshIndex) {

var triangles = mesh.GetTriangles(subMeshIndex);

triNormals[subMeshIndex] = new Vector3[triangles.Length / 3];

for (var i = 0; i < triangles.Length; i += 3) {
int i1 = triangles[i];
int i2 = triangles[i + 1];
int i3 = triangles[i + 2];

// Calculate the normal of the triangle
Vector3 p1 = vertices[i2] - vertices[i1];
Vector3 p2 = vertices[i3] - vertices[i1];
Vector3 normal = Vector3.Cross(p1, p2).normalized;
int triIndex = i / 3;
triNormals[subMeshIndex][triIndex] = normal;

List entry;
VertexKey key;

if (!dictionary.TryGetValue(key = new VertexKey(vertices[i1]), out entry)) {
entry = new List(4);
dictionary.Add(key, entry);
}
entry.Add(new VertexEntry(subMeshIndex, triIndex, i1));

if (!dictionary.TryGetValue(key = new VertexKey(vertices[i2]), out entry)) {
entry = new List();
dictionary.Add(key, entry);
}
entry.Add(new VertexEntry(subMeshIndex, triIndex, i2));

if (!dictionary.TryGetValue(key = new VertexKey(vertices[i3]), out entry)) {
entry = new List();
dictionary.Add(key, entry);
}
entry.Add(new VertexEntry(subMeshIndex, triIndex, i3));
}
}

// Each entry in the dictionary represents a unique vertex position.

foreach (var vertList in dictionary.Values) {
for (var i = 0; i < vertList.Count; ++i) {

var sum = new Vector3();
var lhsEntry = vertList[i];

for (var j = 0; j < vertList.Count; ++j) {
var rhsEntry = vertList[j];

if (lhsEntry.VertexIndex == rhsEntry.VertexIndex) {
sum += triNormals[rhsEntry.MeshIndex][rhsEntry.TriangleIndex];
} else {
// The dot product is the cosine of the angle between the two triangles.
// A larger cosine means a smaller angle.
var dot = Vector3.Dot(
triNormals[lhsEntry.MeshIndex][lhsEntry.TriangleIndex],
triNormals[rhsEntry.MeshIndex][rhsEntry.TriangleIndex]);
if (dot >= cosineThreshold) {
sum += triNormals[rhsEntry.MeshIndex][rhsEntry.TriangleIndex];
}
}
}

normals[lhsEntry.VertexIndex] = sum.normalized;
}
}

mesh.normals = normals;
}

private struct VertexKey
{
private readonly long _x;
private readonly long _y;
private readonly long _z;

// Change this if you require a different precision.
private const int Tolerance = 100000;

// Magic FNV values. Do not change these.
private const long FNV32Init = 0x811c9dc5;
private const long FNV32Prime = 0x01000193;

public VertexKey(Vector3 position) {
_x = (long)(Mathf.Round(position.x * Tolerance));
_y = (long)(Mathf.Round(position.y * Tolerance));
_z = (long)(Mathf.Round(position.z * Tolerance));
}

public override bool Equals(object obj) {
var key = (VertexKey)obj;
return _x == key._x && _y == key._y && _z == key._z;
}

public override int GetHashCode() {
long rv = FNV32Init;
rv ^= _x;
rv *= FNV32Prime;
rv ^= _y;
rv *= FNV32Prime;
rv ^= _z;
rv *= FNV32Prime;

return rv.GetHashCode();
}
}

private struct VertexEntry {
public int MeshIndex;
public int TriangleIndex;
public int VertexIndex;

public VertexEntry(int meshIndex, int triIndex, int vertIndex) {
MeshIndex = meshIndex;
TriangleIndex = triIndex;
VertexIndex = vertIndex;
}
}
}


总数:0 页次:1/0 首页 尾页  
总数:0 页次:1/0 首页 尾页  


所在合集/目录
u3d平滑 更多



发表评论:
文本/html模式切换 插入图片 文本/html模式切换


附件:



NEWBT官方QQ群1: 276678893
可求档连环画,漫画;询问文本处理大师等软件使用技巧;求档softhub软件下载及使用技巧.
但不可"开车",严禁国家敏感话题,不可求档涉及版权的文档软件.
验证问题说明申请入群原因即可.

Copyright © 2005-2020 clq, All Rights Reserved
版权所有
桂ICP备15002303号-1