一、新建空白插件
data:image/s3,"s3://crabby-images/682a0/682a04e1ebad112d7a98546e1ff50d493bb4e7d6" alt=""
在Bulid.cs内加入两个模块,"EditorSubsystem","UnrealEd",
PublicDependencyModuleNames.AddRange(
new string[]
{
"Core",
"EditorSubsystem",
"UnrealEd",
// ... add other public dependencies that you statically link with here ...
}
);
创建继承于EditorSubsystenm的子类
data:image/s3,"s3://crabby-images/dfb8b/dfb8bf5b4f59c24770d55c0a0668bdffa300f095" alt=""
?data:image/s3,"s3://crabby-images/ce3f6/ce3f65dd22fbc5c5173727c6ed5915224fde1e48" alt=""
点创建后会报错,没关系回C++重新编译一下就好
data:image/s3,"s3://crabby-images/21a96/21a96f4e893f7f34681dda6e2f2ef98065c29b03" alt=""
二、C++代码
ExportSixViewsSub.h
// Fill out your copyright notice in the Description page of Project Settings.
#pragma once
#include "CoreMinimal.h"
#include "EditorSubsystem.h"
#include "ExportSixViewsSub.generated.h"
class USpringArmComponent;
/**
*
*/
UCLASS()
class EXPORTSIXVIEWS_API UExportSixViewsSub : public UEditorSubsystem
{
GENERATED_BODY()
public:
//获取World
UFUNCTION(BlueprintPure, Category = "FunTool")
UWorld* GetContextWorld();
//遍历指定路径下所有的StaticMesh
UFUNCTION(BlueprintCallable, Category = "FunTool")
TArray<UStaticMesh*> FindOrLoadAssetsByPath(const FString& FilePath);
//将Actor的Privot设置为Center
UFUNCTION(BlueprintCallable, Category = "FunTool")
bool SetActorPrivotOnCenter(AActor* InActor, bool bWorldSpace /*= false*/, FVector& CenterLocation);
//拍摄模型六视图并保存到指定文件夹
UFUNCTION(BlueprintCallable, Category = "FunTool")
bool SaveRenderTargetToFile(UTextureRenderTarget2D* rt, const FString& Filepath);
//复制指定文件到指定目录
UFUNCTION(BlueprintCallable, Category = "FunTool")
void CopyFile(UStaticMesh* SM, const FString& Filepath);
//导出完成时的消息提示框
UFUNCTION(BlueprintCallable, Category = "FunTool")
void Tips();
//打开文件管理器并返回选择的文件的目录
UFUNCTION(BlueprintCallable, Category = "FunTool")
void OpenFiles(FString& DirectoryName);
};
ExportSixViewsSub.cpp
// Fill out your copyright notice in the Description page of Project Settings.
#include "ExportSixViewsSub.h"
#include "EngineUtils.h"
#include "UnrealEdGlobals.h"
#include "Editor/UnrealEdEngine.h"
#include "Engine/TextureRenderTarget2D.h"
#include "Misc/FileHelper.h"
#include "ImageUtils.h"
#include "GameFramework/SpringArmComponent.h"
#include "Kismet/KismetSystemLibrary.h"
#include "Developer/DesktopPlatform/Public/IDesktopPlatform.h"
#include "Developer/DesktopPlatform/Public/DesktopPlatformModule.h"
UWorld* UExportSixViewsSub::GetContextWorld()
{
UWorld* World = GEditor->GetEditorWorldContext().World();
return World;
}
TArray<UStaticMesh*> UExportSixViewsSub::FindOrLoadAssetsByPath(const FString& FilePath)
{
TArray<UObject*> OutAssets;
TArray<UStaticMesh*> OutStaticMeshs;
FString GamePath = FilePath;
GamePath.ReplaceInline(*FPaths::ProjectContentDir(), TEXT("/Game/"));
if (!GamePath.IsEmpty())
{
EngineUtils::FindOrLoadAssetsByPath(GamePath, OutAssets, EngineUtils::ATL_Regular);
for (auto SM : OutAssets)
{
UStaticMesh* TempSM = Cast<UStaticMesh>(SM);
if (TempSM)
{
OutStaticMeshs.Add(TempSM);
}
}
UE_LOG(LogTemp, Warning, TEXT("FilePaths: Directory Exists"));
return OutStaticMeshs;
}
else
{
UE_LOG(LogTemp, Warning, TEXT("FilePaths: Directory Does not exist"));
return OutStaticMeshs;
}
}
bool UExportSixViewsSub::SetActorPrivotOnCenter(AActor* InActor, bool bWorldSpace /*= false*/, FVector& CenterLocation)
{
if (InActor)
{
FVector PivotOffset = InActor->GetComponentsBoundingBox().GetCenter();
if (bWorldSpace)
{
FVector Delta(PivotOffset - InActor->GetActorLocation());
PivotOffset = InActor->GetTransform().InverseTransformVector(Delta);
}
if (InActor->GetPivotOffset() == PivotOffset)
{
CenterLocation = InActor->GetActorLocation();
return false;
}
else
{
InActor->Modify();
InActor->SetPivotOffset(PivotOffset);
GUnrealEd->SetPivotMovedIndependently(false);
InActor->PostEditMove(/*bFinished=*/ true);
CenterLocation = InActor->GetActorLocation() + PivotOffset;
return true;
}
}
return false;
}
bool UExportSixViewsSub::SaveRenderTargetToFile(UTextureRenderTarget2D* rt, const FString& fileDestination)
{
FTextureRenderTargetResource* rtResource = rt->GameThread_GetRenderTargetResource();
FReadSurfaceDataFlags readPixelFlags(RCM_UNorm);
TArray<FColor> outBMP;
for (FColor& color : outBMP)
{
color.A = 255;
}
outBMP.AddUninitialized(rt->GetSurfaceWidth() * rt->GetSurfaceHeight());
rtResource->ReadPixels(outBMP, readPixelFlags);
FIntPoint destSize(rt->GetSurfaceWidth(), rt->GetSurfaceHeight());
TArray<uint8> CompressedBitmap;
FImageUtils::CompressImageArray(destSize.X, destSize.Y, outBMP, CompressedBitmap);
bool imageSavedOk = FFileHelper::SaveArrayToFile(CompressedBitmap, *fileDestination);
return imageSavedOk;
}
void UExportSixViewsSub::CopyFile(UStaticMesh* SM, const FString& FilePath)
{
FString SM_AbsolutePath = UKismetSystemLibrary::GetSystemPath(SM);
FString FileDestination = FilePath;
FileDestination.Append("/" + SM->GetFName().ToString() + ".uasset");
IPlatformFile& FileManager = FPlatformFileManager::Get().GetPlatformFile();
if (FileManager.CreateDirectory(*FilePath))
{
UE_LOG(LogTemp, Warning, TEXT("FilePaths: Directory was created"));
}
else
{
UE_LOG(LogTemp, Warning, TEXT("FilePaths: Directory was not created"));
}
if (FileManager.CopyFile(*FileDestination, *SM_AbsolutePath))
{
UE_LOG(LogTemp, Warning, TEXT("FilePaths: File Copied!"));
}
else
{
UE_LOG(LogTemp, Warning, TEXT("FilePaths: File not Copied!"));
}
}
void UExportSixViewsSub::Tips()
{
FText const Title = FText::FromString("Message");
FText const DialogText = FText::FromString("Export complete!");
EAppReturnType::Type const ReturnType = FMessageDialog::Open(EAppMsgType::Ok, DialogText, &Title);
}
void UExportSixViewsSub::OpenFiles(FString& DirectoryName)
{
IDesktopPlatform* DesktopPlatform = FDesktopPlatformModule::Get();
DesktopPlatform->OpenDirectoryDialog(nullptr, TEXT("ProjectDir"), FPaths::ConvertRelativePathToFull(FPaths::ProjectDir()), DirectoryName);
}
?三、蓝图代码
?data:image/s3,"s3://crabby-images/349c9/349c9e9c0ad1f7f8a18008d6fb6c8d42bff3a253" alt=""
1.新建Actor蓝图,就只添加一个StaticMeshComponent用于测试用
data:image/s3,"s3://crabby-images/d4417/d441746705a5d773690c7b75382576cea9dbebb6" alt=""
?2.新建相机蓝图,添加一个相机臂和SceneCaptureComponent2D,相机臂取消碰撞,设置CaptureSource默认值
data:image/s3,"s3://crabby-images/3d48b/3d48b694cfb7679b59cfc506f9c264e4d86f1d90" alt=""
data:image/s3,"s3://crabby-images/24b58/24b5877322c5803164c115a871c86da97990781a" alt=""
?3.新建结构体和创建表格
data:image/s3,"s3://crabby-images/3c95b/3c95b45f50933cb8400ddc72aea7d9b7d2699873" alt=""
data:image/s3,"s3://crabby-images/5f02a/5f02a0b555e3a07e456b0f75db330829b61a56c9" alt=""
?4.新建编辑器UI
设计界面:
data:image/s3,"s3://crabby-images/be7cf/be7cfe776363bfa00ccfc492d7dbe837a3d719ce" alt=""
?逻辑界面:
自定义事件:拷贝模型资源文件到指定文件夹
data:image/s3,"s3://crabby-images/2516e/2516e1cdb09f3061754bc17cd2cf412c271a77f3" alt=""
自定义事件:将图片保存到指定文件夹
data:image/s3,"s3://crabby-images/97629/9762942169c080b00a6774c60ad1c9a0efe889e9" alt=""
销毁Actor的函数
data:image/s3,"s3://crabby-images/f6e02/f6e0288417714f91c622266ab7f8b059eb58ab84" alt=""
?Button Pressed事件 选择文件夹
data:image/s3,"s3://crabby-images/0577c/0577c3182a427705975da53937dce56512e74701" alt=""
data:image/s3,"s3://crabby-images/b889b/b889b1694fc2bdbe165bf7bc65e9d01995dd292a" alt=""
?Button Pressed事件 导出图片
data:image/s3,"s3://crabby-images/452ae/452aee8e1b638bb7dbab7804b7cbd231a431b14f" alt=""
data:image/s3,"s3://crabby-images/c9f3c/c9f3c6466ac82713e788ac6d2cf14456ad86761d" alt=""
data:image/s3,"s3://crabby-images/65216/6521627ba65ca47ae42edc02ca56c0f1ede1c086" alt=""
?四、测试
data:image/s3,"s3://crabby-images/ad8b0/ad8b0543a9ecc4bb717059d6f7acaf9a1560581b" alt=""
?data:image/s3,"s3://crabby-images/941f4/941f40f7356d8badb541767289b34eddf35ec915" alt=""
data:image/s3,"s3://crabby-images/68f5c/68f5c9575425cabffb48332dac9fab8c384e183b" alt=""
data:image/s3,"s3://crabby-images/b978c/b978c9be1bf0bd4aeee61ab805cf430df95c5676" alt=""
?五、插件下载(4.27)
链接:https://pan.baidu.com/s/1q7uYrQ41xaY1mIR0ZuQGmA?pwd=rx5y? 提取码:rx5y
|