Files
UnrealEngineUWP/Engine/Plugins/Experimental/MeshModelingToolset/Source/MeshModelingTools/Public/RemoveOccludedTrianglesTool.h
Ryan Schmidt e196c256e4 GeometryProcessing: remove forwarding headers used in GeometryCore transition, and update all affected includes.
#rb none
#rnx
#jira none
#preflight 60c52c5db9446100014da02d

[CL 16653115 by Ryan Schmidt in ue5-main branch]
2021-06-13 00:35:22 -04:00

229 lines
7.5 KiB
C++

// Copyright Epic Games, Inc. All Rights Reserved.
#pragma once
#include "CoreMinimal.h"
#include "UObject/NoExportTypes.h"
#include "MultiSelectionTool.h"
#include "InteractiveToolBuilder.h"
#include "MeshOpPreviewHelpers.h"
#include "CleaningOps/RemoveOccludedTrianglesOp.h"
#include "DynamicMesh/DynamicMesh3.h"
#include "MeshAdapter.h"
#include "BaseTools/SingleClickTool.h"
#include "PropertySets/PolygroupLayersProperties.h"
#include "RemoveOccludedTrianglesTool.generated.h"
PREDECLARE_USE_GEOMETRY_CLASS(FDynamicMesh3);
/**
*
*/
UCLASS()
class MESHMODELINGTOOLS_API URemoveOccludedTrianglesToolBuilder : public UInteractiveToolBuilder
{
GENERATED_BODY()
public:
virtual bool CanBuildTool(const FToolBuilderState& SceneState) const override;
virtual UInteractiveTool* BuildTool(const FToolBuilderState& SceneState) const override;
protected:
virtual const FToolTargetTypeRequirements& GetTargetRequirements() const override;
};
// these UIMode enums are versions of the enums in Operations/RemoveOccludedTriangles.h, w/ some removed & some renamed to be more user friendly
UENUM()
enum class EOcclusionTriangleSamplingUIMode : uint8
{
// currently do not expose centroid-only option; it almost always looks bad
Vertices,
VerticesAndCentroids
};
UENUM()
enum class EOcclusionCalculationUIMode : uint8
{
GeneralizedWindingNumber, // maps to using fast winding number approximation
RaycastOcclusionSamples
};
UENUM()
enum class EOccludedAction : uint8
{
Remove,
SetNewGroup
};
/**
* Standard properties
*/
UCLASS()
class MESHMODELINGTOOLS_API URemoveOccludedTrianglesToolProperties : public UInteractiveToolPropertySet
{
GENERATED_BODY()
public:
URemoveOccludedTrianglesToolProperties();
/** The method for deciding whether a triangle is occluded */
UPROPERTY(EditAnywhere, Category = OcclusionCalculation)
EOcclusionCalculationUIMode OcclusionTestMethod = EOcclusionCalculationUIMode::GeneralizedWindingNumber;
/** Where to sample triangles to test occlusion */
UPROPERTY(EditAnywhere, Category = OcclusionCalculation)
EOcclusionTriangleSamplingUIMode TriangleSampling = EOcclusionTriangleSamplingUIMode::VerticesAndCentroids;
/** The winding isovalue for GeneralizedWindingNumber mode */
UPROPERTY(EditAnywhere, Category = OcclusionCalculation, meta = (UIMin = "-1", UIMax = "1", ClampMin = "-2", ClampMax = "2", EditCondition = "OcclusionTestMethod==EOcclusionCalculationUIMode::GeneralizedWindingNumber"))
double WindingIsoValue = 0.5;
/** For raycast-based occlusion tests, optionally add random ray direction to increase the accuracy of the visibility sampling */
UPROPERTY(EditAnywhere, Category = OcclusionCalculation, meta = (UIMin = "0", UIMax = "100", ClampMin = "0", ClampMax = "1000", EditCondition = "OcclusionTestMethod==EOcclusionCalculationUIMode::RaycastOcclusionSamples"))
int AddRandomRays = 0;
/** Optionally add random samples to each triangle (in addition to those from TriangleSampling) to increase the accuracy of the visibility sampling */
UPROPERTY(EditAnywhere, Category = OcclusionCalculation, meta = (UIMin = "0", UIMax = "100", ClampMin = "0", ClampMax = "1000"))
int AddTriangleSamples = 0;
/** If false, when multiple meshes are selected the meshes can occlude each other. When true, we process each selected mesh independently and only consider self-occlusions. */
UPROPERTY(EditAnywhere, Category = OcclusionCalculation)
bool bOnlySelfOcclude = false;
/** Shrink (erode) the boundary of the set of triangles to remove. */
UPROPERTY(EditAnywhere, Category = OcclusionCalculation, meta = (UIMin = "0", ClampMin = "0"))
int ShrinkRemoval = 0;
UPROPERTY(EditAnywhere, Category = RemoveIslands, meta = (UIMin = "0", ClampMin = "0"))
double MinAreaIsland = 0;
UPROPERTY(EditAnywhere, Category = RemoveIslands, meta = (UIMin = "0", ClampMin = "0"))
int MinTriCountIsland = 0;
/** What action to perform on occluded triangles */
UPROPERTY(EditAnywhere, Category = Action)
EOccludedAction Action = EOccludedAction::Remove;
};
/**
* Advanced properties
*/
UCLASS()
class MESHMODELINGTOOLS_API URemoveOccludedTrianglesAdvancedProperties : public UInteractiveToolPropertySet
{
GENERATED_BODY()
public:
URemoveOccludedTrianglesAdvancedProperties();
/** Amount to numerically 'nudge' occlusion sample query points away from the surface (to avoid e.g. all occlusion sample rays hitting the source triangle) */
// probably not actually a good idea to expose this to the user
//UPROPERTY(EditAnywhere, Category = NormalsTopology, meta = (UIMin = "0.000000001", UIMax = ".0001", ClampMin = "0.0", ClampMax = "0.01"))
double NormalOffset = FMathd::ZeroTolerance;
};
/**
* Factory with enough info to spawn the background-thread Operator to do a chunk of work for the tool
* stores a pointer to the tool and enough info to know which specific operator it should spawn
*/
UCLASS()
class MESHMODELINGTOOLS_API URemoveOccludedTrianglesOperatorFactory : public UObject, public UE::Geometry::IDynamicMeshOperatorFactory
{
GENERATED_BODY()
public:
// IDynamicMeshOperatorFactory API
virtual TUniquePtr<UE::Geometry::FDynamicMeshOperator> MakeNewOperator() override;
UPROPERTY()
URemoveOccludedTrianglesTool *Tool;
int PreviewIdx;
};
/**
* Simple Mesh Normal Updating Tool
*/
UCLASS()
class MESHMODELINGTOOLS_API URemoveOccludedTrianglesTool : public UMultiSelectionTool
{
GENERATED_BODY()
public:
friend URemoveOccludedTrianglesOperatorFactory;
URemoveOccludedTrianglesTool();
virtual void Setup() override;
virtual void Shutdown(EToolShutdownType ShutdownType) override;
virtual void SetWorld(UWorld* World);
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;
#if WITH_EDITOR
virtual void PostEditChangeProperty(FPropertyChangedEvent &PropertyChangedEvent) override;
#endif
virtual void OnPropertyModified(UObject* PropertySet, FProperty* Property) override;
protected:
UPROPERTY()
URemoveOccludedTrianglesToolProperties* BasicProperties;
UPROPERTY()
UPolygroupLayersProperties* PolygroupLayersProperties;
UPROPERTY()
URemoveOccludedTrianglesAdvancedProperties* AdvancedProperties;
UPROPERTY()
TArray<UMeshOpPreviewWithBackgroundCompute*> Previews;
// When multiple meshes in the selection correspond to the same asset, only one needs a PreviewWithBackgroundCompute
// all others just get a plain PreviewMesh copy that is updated via OnMeshUpdated broadcast from the source Preview
UPROPERTY()
TArray<UPreviewMesh*> PreviewCopies;
protected:
TArray<TSharedPtr<FDynamicMesh3, ESPMode::ThreadSafe>> OriginalDynamicMeshes;
// AABB trees and winding trees for every mesh target, with repeated instances as pointers to the same data
TArray<TSharedPtr<UE::Geometry::FDynamicMeshAABBTree3, ESPMode::ThreadSafe>> OccluderTrees;
TArray<TSharedPtr<UE::Geometry::TFastWindingTree<FDynamicMesh3>, ESPMode::ThreadSafe>> OccluderWindings;
TArray<UE::Geometry::FTransform3d> OccluderTransforms;
TArray<TArray<int32>> PreviewToCopyIdx;
TArray<int32> PreviewToTargetIdx;
TArray<int32> TargetToPreviewIdx;
// Group IDs for occluded triangles, per mesh in the Previews array, if OccludedAction is SetNewGroup
TArray<int32> OccludedGroupIDs;
// Selected layer indices for Group IDs, per mesh in the Previews array
TArray<int32> OccludedGroupLayers;
UWorld* TargetWorld;
FViewCameraState CameraState;
void SetupPreviews();
void MakePolygroupLayerProperties();
void GenerateAsset(const TArray<FDynamicMeshOpResult>& Results);
};