Files
UnrealEngineUWP/Engine/Plugins/Experimental/MeshModelingToolsetExp/Source/MeshModelingToolsExp/Public/WeldMeshEdgesTool.h
david hill b3a71a15e2 Mesh Modeling Tools: Improve Weld Mesh Edges tool to optionally merge attributes.
Geometry Core: Adding Split Attribute Welder.

#jira UE-182345
#rb ryan.schmidt
#preflight 643d6fc8211b661dc4f47c6a

[CL 25072473 by david hill in ue5-main branch]
2023-04-17 15:22:49 -04:00

152 lines
5.1 KiB
C++

// Copyright Epic Games, Inc. All Rights Reserved.
#pragma once
#include "CoreMinimal.h"
#include "BaseTools/SingleSelectionMeshEditingTool.h"
#include "PreviewMesh.h"
#include "ModelingOperators.h"
#include "MeshOpPreviewHelpers.h"
#include "DynamicMesh/DynamicMesh3.h"
#include "DynamicMesh/DynamicMeshAABBTree3.h"
#include "WeldMeshEdgesTool.generated.h"
// predeclarations
struct FMeshDescription;
class UMeshElementsVisualizer;
class UWeldMeshEdgesTool;
class FWeldMeshEdgesOp;
/**
*
*/
UCLASS()
class MESHMODELINGTOOLSEXP_API UWeldMeshEdgesToolBuilder : public USingleSelectionMeshEditingToolBuilder
{
GENERATED_BODY()
public:
virtual USingleSelectionMeshEditingTool* CreateNewTool(const FToolBuilderState& SceneState) const override;
};
UENUM()
enum class EWeldMeshEdgesAttributeUIMode : uint8
{
/** Do not weld split-attributes*/
None,
/** Apply attribute welding only along the current mesh welds */
OnWeldedMeshEdgesOnly,
/** Apply attribute welding to all split-attributes */
OnFullMesh
};
UCLASS()
class MESHMODELINGTOOLSEXP_API UWeldMeshEdgesToolProperties : public UInteractiveToolPropertySet
{
GENERATED_BODY()
public:
/** Edges are considered matching if both pairs of endpoint vertices are closer than this distance */
UPROPERTY(EditAnywhere, Category = Options, meta = (UIMin = "0.000001", UIMax = "0.01", ClampMin = "0.00000001", ClampMax = "1000.0"))
float Tolerance = FMathf::ZeroTolerance;
/** Only merge unambiguous pairs that have unique duplicate-edge matches */
UPROPERTY(EditAnywhere, Category = Options)
bool bOnlyUnique = false;
/** If enabled, after an initial attempt at Welding, attempt to resolve remaining open edges in T-junction configurations via edge splits, and then retry Weld */
UPROPERTY(EditAnywhere, Category = Options)
bool bResolveTJunctions = false;
/** Initial number of open boundary edges */
UPROPERTY(VisibleAnywhere, Category = Statistics)
int32 InitialEdges;
/** Number of remaining open boundary edges */
UPROPERTY(VisibleAnywhere, Category = Statistics)
int32 RemainingEdges;
/** Controls split-attribute welding performed after the Mesh weld. Applies to normals, tangents, UVs and colors. */
UPROPERTY(EditAnywhere, Category = Options, meta = (DisplayName = "Split-Attribute Welding"))
EWeldMeshEdgesAttributeUIMode AttrWeldingMode = EWeldMeshEdgesAttributeUIMode::OnWeldedMeshEdgesOnly;
/** Threshold on the angle between normals used to determine if split normals should be merged*/
UPROPERTY(EditAnywhere, Category = Options, AdvancedDisplay, meta = (EditCondition = "AttrWeldingMode != EWeldMeshEdgesAttributeUIMode::None", UIMin = "0.0", UIMax = "180.0", ClampMin = "0.0", ClampMax = "180.0", DisplayName="Normal Angle Threshold"))
float SplitNormalThreshold = 0.1f;
/** Threshold on the angle between tangent used to determine if split tangents should be merged*/
UPROPERTY(EditAnywhere, Category = Options, AdvancedDisplay, meta = (EditCondition = "AttrWeldingMode != EWeldMeshEdgesAttributeUIMode::None", UIMin = "0.0", UIMax = "180.0", ClampMin = "0.0", ClampMax = "180.0", DisplayName="Tangent Angle Threshold"))
float SplitTangentsThreshold = 0.1f;
/** Threshold uv-distance used to determine if split UVs should be merged */
UPROPERTY(EditAnywhere, Category = Options, AdvancedDisplay, meta = (EditCondition = "AttrWeldingMode != EWeldMeshEdgesAttributeUIMode::None", UIMin = "0" , DisplayName = "UV Distance Threshold"))
float SplitUVThreshold = 0.01f;
/** Threshold color-distance used to determine if split colors should be merged */
UPROPERTY(EditAnywhere, Category = Options, AdvancedDisplay, meta = (EditCondition = "AttrWeldingMode != EWeldMeshEdgesAttributeUIMode::None", UIMin = "0", DisplayName = "Color Distance Threshold"))
float SplitColorThreshold = 0.01f;
};
UCLASS()
class MESHMODELINGTOOLSEXP_API UWeldMeshEdgesOperatorFactory : public UObject, public UE::Geometry::IDynamicMeshOperatorFactory
{
GENERATED_BODY()
public:
// IDynamicMeshOperatorFactory API
virtual TUniquePtr<UE::Geometry::FDynamicMeshOperator> MakeNewOperator() override;
UPROPERTY()
TObjectPtr<UWeldMeshEdgesTool> WeldMeshEdgesTool;
};
/**
* Mesh Weld Edges Tool
*/
UCLASS()
class MESHMODELINGTOOLSEXP_API UWeldMeshEdgesTool : public USingleSelectionMeshEditingTool
{
GENERATED_BODY()
public:
UWeldMeshEdgesTool();
virtual void Setup() override;
virtual void OnShutdown(EToolShutdownType ShutdownType) override;
virtual void OnTick(float DeltaTime) override;
virtual bool HasCancel() const override { return true; }
virtual bool HasAccept() const override { return true; }
virtual bool CanAccept() const override;
// update parameters in FWeldMeshEdgesOp based on current Settings
void UpdateOpParameters(FWeldMeshEdgesOp& Op) const;
protected:
UPROPERTY()
TObjectPtr<UWeldMeshEdgesToolProperties> Settings;
UPROPERTY()
TObjectPtr<UMeshOpPreviewWithBackgroundCompute> PreviewCompute = nullptr;
UPROPERTY()
TObjectPtr<UMeshElementsVisualizer> MeshElementsDisplay = nullptr;
UPROPERTY()
TObjectPtr<UWeldMeshEdgesOperatorFactory> OperatorFactory;
protected:
TSharedPtr<UE::Geometry::FDynamicMesh3, ESPMode::ThreadSafe> SourceMesh;
};