UniVRM/Assets/VRM10/vrmlib/Runtime/VertexBuffer.cs
2021-10-06 20:44:02 +09:00

291 lines
8.1 KiB
C#

using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using System.Numerics;
using System.Runtime.InteropServices;
using UniGLTF;
namespace VrmLib
{
public class VertexBuffer : IEnumerable<KeyValuePair<string, BufferAccessor>>
{
public Dictionary<string, BufferAccessor> VertexBuffers = new Dictionary<string, BufferAccessor>();
public bool ContainsKey(string key)
{
return VertexBuffers.ContainsKey(key);
}
public IEnumerator<KeyValuePair<string, BufferAccessor>> GetEnumerator()
{
return VertexBuffers.GetEnumerator();
}
IEnumerator IEnumerable.GetEnumerator()
{
return GetEnumerator();
}
public void Add(string key, BufferAccessor accessor)
{
VertexBuffers.Add(key, accessor);
}
public bool TryGetValue(string key, out BufferAccessor accessor)
{
return VertexBuffers.TryGetValue(key, out accessor);
}
public int Count
{
get
{
if (VertexBuffers.TryGetValue(PositionKey, out BufferAccessor buffer))
{
return buffer.Count;
}
return 0;
}
}
public const string PositionKey = "POSITION";
public BufferAccessor Positions
{
get
{
if (VertexBuffers.TryGetValue(PositionKey, out BufferAccessor buffer))
{
return buffer;
}
return null;
}
}
public const string NormalKey = "NORMAL";
public BufferAccessor Normals
{
get
{
if (VertexBuffers.TryGetValue(NormalKey, out BufferAccessor buffer))
{
return buffer;
}
return null;
}
}
public const string TangentKey = "TANGENT";
public const string ColorKey = "COLOR_0";
public BufferAccessor Colors
{
get
{
if (VertexBuffers.TryGetValue(ColorKey, out BufferAccessor buffer))
{
return buffer;
}
return null;
}
}
public const string TexCoordKey = "TEXCOORD_0";
public BufferAccessor TexCoords
{
get
{
if (VertexBuffers.TryGetValue(TexCoordKey, out BufferAccessor buffer))
{
return buffer;
}
return null;
}
}
public const string TexCoordKey2 = "TEXCOORD_1";
public const string JointKey = "JOINTS_0";
public BufferAccessor Joints
{
get
{
if (VertexBuffers.TryGetValue(JointKey, out BufferAccessor buffer))
{
return buffer;
}
return null;
}
}
public const string WeightKey = "WEIGHTS_0";
public BufferAccessor Weights
{
get
{
if (VertexBuffers.TryGetValue(WeightKey, out BufferAccessor buffer))
{
return buffer;
}
return null;
}
}
public void RemoveTangent()
{
if (VertexBuffers.ContainsKey(TangentKey))
{
VertexBuffers.Remove(TangentKey);
}
}
public int ByteLength
{
get
{
return VertexBuffers.Sum(x => x.Value.ByteLength);
}
}
public void ValidateLength(string name = "")
{
foreach (var kv in VertexBuffers)
{
if (kv.Key == PositionKey) continue;
if (kv.Value.Count != Count)
{
var msg = "vertex attribute not same length";
if (!string.IsNullOrEmpty(name))
{
msg = $"{name}: {msg}";
}
throw new ArgumentException(msg);
}
}
}
public void ValidateNAN()
{
foreach (var kv in VertexBuffers)
{
if (kv.Value.ComponentType == AccessorValueType.FLOAT)
{
var values = kv.Value.GetSpan<float>(false);
int i = 0;
foreach (var f in values)
{
if (float.IsNaN(f)) throw new ArithmeticException("float error");
++i;
}
}
}
}
public VertexBuffer()
{
}
public VertexBuffer CloneWithOffset(int offsetCount)
{
var vb = new VertexBuffer();
foreach (var kv in VertexBuffers)
{
vb.VertexBuffers[kv.Key] = kv.Value.CloneWithOffset(offsetCount);
}
return vb;
}
public SpanLike<SkinJoints> GetOrCreateJoints()
{
var buffer = Joints;
if (buffer == null)
{
buffer = new BufferAccessor(
new ArraySegment<byte>(new byte[Marshal.SizeOf(typeof(SkinJoints)) * Count]),
AccessorValueType.UNSIGNED_SHORT,
AccessorVectorType.VEC4, Count);
Add(JointKey, buffer);
}
return SpanLike.Wrap<SkinJoints>(buffer.Bytes);
}
public SpanLike<Vector4> GetOrCreateWeights()
{
var buffer = Weights;
if (buffer == null)
{
buffer = new BufferAccessor(
new ArraySegment<byte>(new byte[Marshal.SizeOf(typeof(Vector4)) * Count]),
AccessorValueType.FLOAT,
AccessorVectorType.VEC4, Count);
Add(WeightKey, buffer);
}
return SpanLike.Wrap<Vector4>(buffer.Bytes);
}
static bool HasSameKeys<T>(Dictionary<string, T> lhs, Dictionary<string, T> rhs)
{
if (lhs.Count != rhs.Count) return false;
foreach (var (l, r) in Enumerable.Zip(lhs.Keys.OrderBy(x => x), rhs.Keys.OrderBy(x => x), (l, r) => (l, r)))
{
if (l != r)
{
return false;
}
}
return true;
}
public void Append(VertexBuffer v)
{
var keys = VertexBuffers.Keys.ToList();
var lastCount = Count;
// v から VertexBufferfs に足す
foreach (var kv in v.VertexBuffers)
{
if (VertexBuffers.TryGetValue(kv.Key, out BufferAccessor buffer))
{
// used
keys.Remove(kv.Key);
if (buffer.Count != lastCount)
{
throw new ArgumentException();
}
}
else
{
// add empty
var byteLength = lastCount * kv.Value.Stride;
buffer = new BufferAccessor(new ArraySegment<byte>(new byte[byteLength]), kv.Value.ComponentType, kv.Value.AccessorType, lastCount);
if (buffer.Count != lastCount)
{
throw new ArgumentException();
}
VertexBuffers.Add(kv.Key, buffer);
}
buffer.Append(kv.Value);
}
// 足されなかったキーに同じ長さを詰める
foreach (var key in keys)
{
var dst = VertexBuffers[key];
dst.Extend(v.Positions.Count);
}
ValidateLength();
}
public void Resize(int n)
{
foreach (var kv in VertexBuffers)
{
kv.Value.Resize(n);
}
}
}
}