183 lines
6.4 KiB
C#
183 lines
6.4 KiB
C#
// Magica Cloth 2.
|
|
// Copyright (c) 2023 MagicaSoft.
|
|
// https://magicasoft.jp
|
|
using Unity.Burst;
|
|
using Unity.Collections;
|
|
using Unity.Jobs;
|
|
using Unity.Mathematics;
|
|
|
|
namespace MagicaCloth2
|
|
{
|
|
/// <summary>
|
|
/// 単純な頂点間の距離によるリダクション
|
|
/// このリダクションは頂点の接続状態を無視する
|
|
/// </summary>
|
|
public class SimpleDistanceReduction : StepReductionBase
|
|
{
|
|
/// <summary>
|
|
/// グリッドマップ
|
|
/// </summary>
|
|
private GridMap<int> gridMap;
|
|
|
|
//=========================================================================================
|
|
public SimpleDistanceReduction(
|
|
string name,
|
|
VirtualMesh mesh,
|
|
ReductionWorkData workingData,
|
|
float startMergeLength,
|
|
float endMergeLength,
|
|
int maxStep,
|
|
bool dontMakeLine,
|
|
float joinPositionAdjustment
|
|
)
|
|
: base($"SimpleDistanceReduction [{name}]", mesh, workingData, startMergeLength, endMergeLength, maxStep, dontMakeLine, joinPositionAdjustment)
|
|
{
|
|
}
|
|
|
|
public override void Dispose()
|
|
{
|
|
base.Dispose();
|
|
gridMap.Dispose();
|
|
}
|
|
|
|
protected override void StepInitialize()
|
|
{
|
|
base.StepInitialize();
|
|
gridMap = new GridMap<int>(vmesh.VertexCount);
|
|
}
|
|
|
|
protected override void CustomReductionStep()
|
|
{
|
|
// 最適なグリッドサイズを割り出す(nowMergeLengthは>0が保証されている)
|
|
float gridSize = nowMergeLength * 2.0f; // 1.5?
|
|
|
|
// 作業用バッファクリア
|
|
gridMap.GetMultiHashMap().Clear();
|
|
|
|
// ポイントをグリッドに登録
|
|
var initGridJob = new InitGridJob()
|
|
{
|
|
vcnt = vmesh.VertexCount,
|
|
gridSize = gridSize,
|
|
localPositions = vmesh.localPositions.GetNativeArray(),
|
|
joinIndices = workData.vertexJoinIndices,
|
|
gridMap = gridMap.GetMultiHashMap(),
|
|
};
|
|
initGridJob.Run();
|
|
|
|
// 近傍ポイントを検索、結合エッジリストに登録
|
|
var searchJob = new SearchJoinEdgeJob()
|
|
{
|
|
vcnt = vmesh.VertexCount,
|
|
gridSize = gridSize,
|
|
radius = nowMergeLength,
|
|
localPositions = vmesh.localPositions.GetNativeArray(),
|
|
joinIndices = workData.vertexJoinIndices,
|
|
vertexToVertexMap = workData.vertexToVertexMap, // cost用
|
|
gridMap = gridMap.GetMultiHashMap(),
|
|
joinEdgeList = joinEdgeList,
|
|
};
|
|
searchJob.Run();
|
|
}
|
|
|
|
[BurstCompile]
|
|
struct InitGridJob : IJob
|
|
{
|
|
public int vcnt;
|
|
public float gridSize;
|
|
|
|
[Unity.Collections.ReadOnly]
|
|
public NativeArray<float3> localPositions;
|
|
[Unity.Collections.ReadOnly]
|
|
public NativeArray<int> joinIndices;
|
|
|
|
public NativeParallelMultiHashMap<int3, int> gridMap;
|
|
|
|
// 頂点ごと
|
|
public void Execute()
|
|
{
|
|
for (int vindex = 0; vindex < vcnt; vindex++)
|
|
{
|
|
if (joinIndices[vindex] >= 0)
|
|
continue; // isDelete
|
|
|
|
GridMap<int>.AddGrid(localPositions[vindex], vindex, gridMap, gridSize);
|
|
}
|
|
}
|
|
}
|
|
|
|
[BurstCompile]
|
|
struct SearchJoinEdgeJob : IJob
|
|
{
|
|
public int vcnt;
|
|
public float gridSize;
|
|
public float radius;
|
|
|
|
[Unity.Collections.ReadOnly]
|
|
public NativeArray<float3> localPositions;
|
|
[Unity.Collections.ReadOnly]
|
|
public NativeArray<int> joinIndices;
|
|
|
|
[Unity.Collections.ReadOnly]
|
|
public NativeParallelMultiHashMap<ushort, ushort> vertexToVertexMap;
|
|
[Unity.Collections.ReadOnly]
|
|
public NativeParallelMultiHashMap<int3, int> gridMap;
|
|
|
|
[Unity.Collections.WriteOnly]
|
|
public NativeList<JoinEdge> joinEdgeList;
|
|
|
|
// 頂点ごと
|
|
public void Execute()
|
|
{
|
|
for (int vindex = 0; vindex < vcnt; vindex++)
|
|
{
|
|
if (joinIndices[vindex] >= 0)
|
|
continue; // isDelete
|
|
|
|
float3 pos = localPositions[vindex];
|
|
|
|
// 自身の接続頂点数
|
|
float linkCount = math.max(vertexToVertexMap.CountValuesForKey((ushort)vindex) - 1, 1);
|
|
|
|
// 範囲グリッド走査
|
|
foreach (int3 grid in GridMap<int>.GetArea(pos, radius, gridMap, gridSize))
|
|
{
|
|
if (gridMap.ContainsKey(grid) == false)
|
|
continue;
|
|
|
|
// このグリッドを検索する
|
|
foreach (int tindex in gridMap.GetValuesForKey(grid))
|
|
{
|
|
// 自身は弾く
|
|
if (tindex == vindex)
|
|
continue;
|
|
|
|
// 距離判定
|
|
float3 tpos = localPositions[tindex];
|
|
float dist = math.distance(pos, tpos);
|
|
if (dist > radius)
|
|
continue;
|
|
|
|
// 相手の接続頂点数
|
|
float tlinkCount = math.max(vertexToVertexMap.CountValuesForKey((ushort)tindex) - 1, 1);
|
|
|
|
// コスト計算
|
|
//float cost = dist / (linkCount + tlinkCount); // todo:とりあえずテスト
|
|
//float cost = dist * (1.0f / (linkCount + tlinkCount));
|
|
float cost = dist * (1.0f + (linkCount + tlinkCount) / 2.0f); // 12
|
|
|
|
// 全部登録
|
|
var pair = new JoinEdge()
|
|
{
|
|
vertexPair = new int2(vindex, tindex),
|
|
cost = cost,
|
|
};
|
|
joinEdgeList.Add(pair);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|