Unreal Engine 4 Scripting with C++ Cookbook 简单笔记
- 主动删除Object 内存 调用 object->ConditionalBeginDestroy() 间隔一段时间后会被垃圾回收系统清除 也可以手动调用GetWorld()->ForceGarbageCollection(true) 来立即清空
时间间隔的设置默认是60s 位置在Epic Games\4.11\Engine\Config \BaseEngine.ini gc.TimeBetweenPurgingPendingKillObjects=60
2.创建UStruct
创建。h文件
pragma once
include "ColoredTexture.generated.h"
USTRUCT()
struct FColoredTexture
{
GENERATED_BODY()
public:
UPROPERTY(EditAnywhere, BlueprintReadWrite, Category = HUD)
UTexture* Texture;
UPROPERTY(EditAnywhere, BlueprintReadWrite, Category = HUD)
FLinearColor Color;
};
3.创建Enum
UENUM()
enum Status
{
Stopped UMETA(DisplayName = "display Stopped"),
Moving,
Attacking
};
使用Enum
UPROPERTY(EditAnywhere, BlueprintReadWrite, Category = "Status")
TEnumAsByte<Status> status;
4.内存管理
继承自UObject的类型通过声明UProperty 保存引用 会自动添加到GC系统中 可以自动删除 或者通过AddToRoot 防止被自动删除 需要删除时 调用RemoveFromRoot自动被gc系统删除
Actor 与 ActorComponent例外
非UObject类可以通过TSharedRef 或者 TWeakRef 智能指针来保存引用 智能指针智能不能保存UObject类型
TWeakPtr 不会保证引用会在内存中保存 所有可以通过ptr.isValid() 来判断指针指向的对象是否存在
TSharePtr 是线程安全的 不需要的画可以使用TAutoPtr
TScopedPointer 作用域智能指针 确保指针会在作用域结束后才能删除
如果使用TArray之类的容器保存UObject指针 需要将TArray声明UPorperty()
5.控制Actor的生存时间 一种方式是通过调用延迟函数 在延迟回调里面通过actor->destroy() 破坏掉
GetWorldTimerManager().SetTimer(Timer, this,
&AUE4CookbookGameMode::DestroyActorFunction, 10)
还有一种方式是通过 设置setLifeSpan(10) 在10s后actor会调用自身的Destroy()删除
- 没有任何Component的actor是没有任何意义的 既没有transform也不能附加到其他的actor上
为Actor添加Component的话 通过在Actor的构造函数(必须要在构造函数中)调用CreateDefaultSubobject 创建Component 测试发现当将CreateDefaultSubobject创建的
Component保存为UProperty是 名称参数不起作用 实际在Editor中显示的名称与变量名称一致
加载StaticMesh资源 auto meshAsset = ConstructorHelpers::FObjectFinder<UStaticMesh>(TEXT("/Game/StarterContent/Props/SM_Bush"));
其中路径部分的game对应的是Content文件夹 可以通过在资源右键 copyreference 获得该资源的路径
7.可以通过继承自ActorComponent 重写component类 可以通过GetOwner()获得Component绑定的Actor
UCLASS( ClassGroup=(Custom), meta=(BlueprintSpawnableComponent) ) 其中ClassGroup指定 在Editor中 component的分类 meta 指定该Component是否可以被蓝图actor使用
- 代理与事件的区别 代理的调用可以在任何可以访问到代理的类中进行访问 而事件 只可以在声明事件的类中进行访问
9.在代码中控制输出 继承Character类 在SetupPlayerInputComponent(UInputComponent* PlayerInputComponent)
{中 调用PlayerInputComponent->BindAxis() 将在ProjectSetting、input中的动作名称 绑定到自定义的函数上
同理可调用BindAction() 绑定动作按键
在代码中动态的添加按键动作
创建AxisKey FInputAxisKeyMapping forwardKey("Forward", EKeys::W, 1.0f)
添加key GetWorld()->GetFirstPlayerController()->PlayerInput->AddAxisMapping(forwardKey)创建接口类
接口类指的是一组classs声明 其中U开头的类 继承自UInterface I开头的类是声明实际接口方法的类
创建MyInterface。h 写出一下代码
// Fill out your copyright notice in the Description page of Project Settings.
pragma once
include "CoreMinimal.h"
include "MyInterface.generated.h"
/**
*/
UINTERFACE()
class UE4COOK_API UMyInterface:public UInterface
{
GENERATED_BODY()
};
class UE4COOK_API IMyInterface
{
GENERATED_BODY()
public:
virtual FString GetTestName();
};
创建MyInterface.cpp
// Fill out your copyright notice in the Description page of Project Settings.
include "MyInterface.h"
include "UE4Cook.h"
FString IMyInterface::GetTestName()
{
unimplemented();
return FString();
}
接口的使用 创建一个新的actor类 继承该接口 并重写GetTestName类
class UE4COOK_API ASingleInterfaceActor : public AActor, public IMyInterface
{
GENERATED_BODY()
public:
// Sets default values for this actor's properties
ASingleInterfaceActor();
protected:
// Called when the game starts or when spawned
virtual void BeginPlay() override;
public:
// Called every frame
virtual void Tick(float DeltaTime) override;
FString GetTestName() override;
};
检测是否实现接口 可通过
obj->GetClass()->ImplementsInterface(UInterface::StaticClass());
如果要创建继承的接口类 就要以UMyInterface为父类 IMyInterface为父类 按照上面的格式 重新添加接口类
12.创建一个被editor 蓝图识别的UObject类型
需要在UClass中 添加BlueprintType 表示该类可以被蓝图认为是一种type类型 可以作为返回 和 输入的节点
- 添加module的流程
右键.uproject文件 添加新的模块
{
"FileVersion": 3,
"EngineAssociation": "4.16",
"Category": "",
"Description": "",
"Modules": [
{
"Name": "UE4Cook",
"Type": "Runtime",
"LoadingPhase": "Default",
"AdditionalDependencies": [
"Engine",
"CoreUObject"
]
},
{
"Name": "UE4CookTestEditor", //新模块名称
"Type": "Editor", //运行模式 Runtime 表示 既在editor模式下运行 又在发布版运行
"LoadingPhase": "PostEngineInit", //模块加载时机
"AdditionalDependencies": [
"Engine",
"CoreUObject"
]
}
]
}
添加配置文件
在Source下 参照原有的文件接口 添加UE4CookTestEditor 文件夹 里面相应创建xx.h xx.build.cs xx.cpp文件
.build.cs
using UnrealBuildTool;
public class UE4CookTestEditor : ModuleRules
{
public UE4CookTestEditor(ReadOnlyTargetRules Target) : base(Target)
{
PublicDependencyModuleNames.AddRange(new string[] {"Core", "CoreUObject", "Engine", "InputCore", "RHI","RenderCore", "ShaderCore" });
PublicDependencyModuleNames.Add("UE4Cook"); //主模块
PrivateDependencyModuleNames.AddRange(new string[] {"UnrealEd" });
}
}
其中。h文件
#pragma once
include "CoreMinimal.h"
include "Engine.h"
include "ModuleManager.h"
include "UnrealEd.h" //为了使用Editor相关的函数 可以不加
class FUE4CookTestEditorModule: public IModuleInterface
{
};
.cpp文件
include "UE4CookTestEditor.h"
include "Modules/ModuleManager.h"
IMPLEMENT_PRIMARY_GAME_MODULE( FUE4CookTestEditorModule, UE4CookTestEditor); //此处的名称要与。uproject中配置的名称保持一致
添加完以上文件后 右键。uproject 重新生成vsstudio文件
编辑UE4CookTestEditor.Target.cs
using UnrealBuildTool;
using System.Collections.Generic;
public class UE4CookEditorTarget : TargetRules
{
public UE4CookEditorTarget(TargetInfo Target) : base(Target)
{
Type = TargetType.Editor;
ExtraModuleNames.AddRange( new string[] { "UE4CookTestEditor" } );
}
}
编译程序并运行 点开/developTools/Modules 查看自定义的module是否正确添加
主要要点 一定要确保模块名称在文件名 。build.cs .h 。cpp中的一致性 模块创建失败多是由于名称不一致引起的
- 添加工具ui到Editor中
editor的ui是通过TCommand的方式进行声明
在。build.cs中
publicDependencyModuleNames 添加Slate模块
添加CookbookCommands.h
pragma once
include "Commands.h"
include "EditorStyleSet.h"
class FCookbookCommands : public TCommands<FCookbookCommands>
{
public:
FCookbookCommands() :TCommands<FCookbookCommands>(FName(TEXT("UE4_Cookbook")), //command 名称
FText::FromString("Cookbook Commands"), // 提示信息
NAME_None, //配合提示信息的 参数
FEditorStyle::GetStyleSetName())
{};
virtual void RegisterCommands() override;
TSharedPtr<FUICommandInfo> myButton;
};
添加CookbookCommands.cpp
include "UE4CookTestEditor.h"
include "Commands.h"
include "CookbookCommands.h"
void FCookbookCommands::RegisterCommands()
{
define LOCTEXT_NAMESPACE ""
UI_COMMAND(myButton, "Cookbook", "Demo Cookbook Toolbar command", EUserInterfaceActionType::Button, FInputGesture()); //创建button 保存到myButton中
undef LOCTEXT_NAMESPACE
}
在Editor module类中 重写StartupModule ShutdownModule函数 在其中将之前定义的command 绑定到具体的事件和ui中
editor.h文件
// Fill out your copyright notice in the Description page of Project Settings.
pragma once
include "CoreMinimal.h"
include "Engine.h"
include "ModuleManager.h"
include "UnrealEd.h"
include "MainFrame.h" //需要在。build.cs中添加Mainframe模块
include "CookbookCommands.h"
include "MultiBoxExtender.h"
class FUE4CookTestEditorModule: public IModuleInterface
{
public:
virtual void StartupModule() override;
virtual void ShutdownModule() override;
TSharedPtr<FExtender> ToolbarExtender; //菜单类 可以用来拓展右键菜单 和工具栏菜单
TSharedPtr<const FExtensionBase> Extension; //拓展类 存储添加菜单的结果
//按钮点击事件
void MyButton_Clicked()
{
TSharedPtr<SWindow> CookbookWindow = SNew(SWindow)
.Title(FText::FromString(TEXT("Co Window")))
.ClientSize(FVector2D(800, 400))
.SupportsMaximize(false)
.SupportsMinimize(false);
IMainFrameModule& mainFrameModule = FModuleManager::LoadModuleChecked<IMainFrameModule>(TEXT("MainFrame"));
if (mainFrameModule.GetParentWindow().IsValid())
{
FSlateApplication::Get().AddWindowAsNativeChild(CookbookWindow, mainFrameModule.GetParentWindow().ToSharedRef())
}
else
{
FSlateApplication::Get().AddWindow(CookbookWindow);
}
};
//添加到工具栏的按钮添加执行函数
void AddToolbarExtension(FToolBarBuilder& builder)
{
//创建按钮图标
FSlateIcon iconBrush = FSlateIcon(FEditorStyle::GetStyleSetName(), "LevelEditor.ViewOptions", "LevelEditor.ViewOptions.Small");
//添加按钮 第一个参数是
builder.AddToolBarButton(FCookbookCommands::Get().myButton, NAME_None, FText::FromString("btn"), FText::FromString("Click to display", iconBrush, NAME_None));
};
};
editor.cpp文件
// Fill out your copyright notice in the Description page of Project Settings.
include "UE4CookTestEditor.h"
include "Modules/ModuleManager.h"
include "ILevelEditor.h"
include "SlateBasics.h"
IMPLEMENT_GAME_MODULE(FUE4CookTestEditorModule, UE4CookTestEditor);
void FUE4CookTestEditorModule::StartupModule()
{
FCookbookCommands::Register(); //注册自定义command
TSharedPtr<FUICommandList> CommandList = MakeShareable(new FUICommandList()); //创建commandlist 保存我们绑定的事件
//绑定按钮事件 第一个参数是我们创建的按钮 第二个参数是按钮的执行回调 第三个参数是执行前的判断回调 为true的画才可以执行按钮事件 这里使用预定义的回调 始终返回true
CommandList->MapAction(FCookbookCommands::Get().myButton, FExecuteAction::CreateRaw(this, &FUE4CookTestEditorModule::MyButton_Clicked), FCanExecuteAction());
ToolbarExtender = MakeShareable(new FExtender()); //创建菜单类 可以用来拓展右键菜单 和工具栏菜单
//添加按钮到工具栏 第一个参数是添加的位置 可以通过Editor Setting/ Generay / miscellaneous /display uiextension points 获得菜单的位置 第二个参数是具体的位置在前
Extension = ToolbarExtender->AddToolBarExtension("Compile", EExtensionHook::Before, CommandList, FToolBarExtensionDelegate::CreateRaw(this,
&FUE4CookTestEditorModule::AddToolbarExtension));
//将工具栏添加到editor中
FLevelEditorModule& LevelEditorModule = FModuleManager::LoadModuleChecked<FLevelEditorModule>("LevelEditor");
LevelEditorModule.GetToolBarExtensibilityManager()->AddExtender(ToolbarExtender);
}
void FUE4CookTestEditorModule::ShutdownModelu()
{
//模块关闭时 移除拓展
ToolbarExtender->RemoveExtension(Extension.ToSharedRef());
Extension.Reset();
ToolbarExtender.Reset();
}
.build.cs
// Fill out your copyright notice in the Description page of Project Settings.
using UnrealBuildTool;
public class UE4CookTestEditor : ModuleRules
{
public UE4CookTestEditor(ReadOnlyTargetRules Target) : base(Target)
{
PublicDependencyModuleNames.AddRange(new string[] { "Core", "CoreUObject", "Engine", "InputCore", "RHI", "RenderCore", "ShaderCore", "Slate", "SlateCore", "MainFrame", "EditorStyle", });
PublicDependencyModuleNames.Add("UE4Cook");
PrivateDependencyModuleNames.AddRange(new string[] { "UnrealEd" });
}
}
添加菜单按钮
与普通按钮不一样的地方
1.在。h的Add函数中 调用builder.AddMenuEntry
2.在。cpp extender的extender->AddMenuExtension 变成AddMenuExtension
3.LevelEditorModule.GetMenuExtensibilityManager()->AddExtender(ToolbarExtender);
添加窗口
可以通过windows/develop tools/widget reflector 点击PickLiveWidget 获得窗口层级
TSharedRef<SWindow> CookbookWindow = SNew(SWindow) //创建一个SWindow类型的Slate
.Title(FText::FromString(TEXT("Cookbook Window"))) //设置window的属性
.ClientSize(FVector2D(800, 400))
.SupportsMaximize(false)
.SupportsMinimize(false)
[ //在【】中添加要拜访到该slot的ui window只有能添加一个widget
SNew(SVerticalBox)
+ SVerticalBox::Slot() //通过重载运算符 + 获得一个slot
.HAlign(HAlign_Center)
.VAlign(VAlign_Center)
[
SNew(STextBlock)
.Text(FText::FromString(TEXT("Hello from Slate")))
]
];
IMainFrameModule& mainFrameModule = FModuleManager::LoadModuleChecked<IMainFrameModule>(TEXT("MainFrame"));
if (mainFrameModule.GetParentWindow().IsValid())
{
FSlateApplication::Get().AddWindowAsNativeChild(CookbookWindow, mainFrameModule.GetParentWindow().ToSharedRef());
}
else
{
FSlateApplication::Get().AddWindow(CookbookWindow);
}
15 创建ue4可用的新的资源类型
创建继承自UObject 自定义类
创建继承自UFactory 的自定义类 重写其中的FactoryCreateNew方法 在方法中创建我们自定义的类型
自定义的类型 可以通过在content中 右键 miscellaneous中查询到
。cpp文件
// Fill out your copyright notice in the Description page of Project Settings.
include "CustomAssetFactory.h"
include "MyCustomAsset.h"
UCustomAssetFactory::UCustomAssetFactory():Super()
{
bCreateNew = true;
bEditAfterNew = true;
SupportedClass = UMyCustomAsset::StaticClass();
}
UObject* UCustomAssetFactory::FactoryCreateNew(UClass* InClass, UObject* InParent, FName InName, EObjectFlags Flags, UObject* Context, FFeedbackContext* Warn, FName CallingContext)
{
auto NewObjectAsset = NewObject<UMyCustomAsset>(InParent, InClass, InName, Flags);
return NewObjectAsset;
}
16 创建自定义资源的右键菜单
创建菜单类 继承自FAssetTypeActions_Base
.h文件
pragma once
include "AssetTypeActions_Base.h"
class FMyCustomAssetActions :public FAssetTypeActions_Base
{
public:
virtual bool HasActions(const TArray<UObject>& InObjects)const override; //系统调用 确认该类型是否有action
virtual void GetActions(const TArray<UObject>& InObjects, FMenuBuilder& menuBuilder) override; //如果上面返回true 那么调用下面 添加菜单项
virtual FText GetName() const override; //资源在缩略图模式下 鼠标的tip提示
virtual UClass* GetSupportedClass() const override; //资源缩略图的颜色
virtual FColor GetTypeColor() const override;
virtual uint32 GetCategories() override;
void MyCustomAssetContent_Clicked();
};
.cpp文件
include "UE4CookTestEditor.h"
include "MyCustomAssetActions.h"
include "MyCustomAsset.h"
bool FMyCustomAssetActions::HasActions(const TArray<UObject*>& InObjects) const
{
return true;
}
void FMyCustomAssetActions::GetActions(const TArray<UObject*>& InObjects, FMenuBuilder& menuBuilder)
{
menuBuilder.AddMenuEntry(FText::FromString("CustomAssetAction"),
FText::FromString("Action from Cook"),
FSlateIcon(FEditorStyle::GetStyleSetName(), "LevelEditor.ViewOptions"),
FUIAction(FExecuteAction::CreateRaw(this, &FMyCustomAssetActions::MyCustomAssetContent_Clicked),
FCanExecuteAction())
);
}
uint32 FMyCustomAssetActions::GetCategories()
{
return EAssetTypeCategories::Misc;
}
FText FMyCustomAssetActions::GetName() const
{
return FText::FromString(TEXT("My Custom Asset"));
}
UClass* FMyCustomAssetActions::GetSupportedClass() const
{
return UMyCustomAsset::StaticClass();
}
FColor FMyCustomAssetActions::GetTypeColor() const
{
return FColor::Emerald;
}
void FMyCustomAssetActions::MyCustomAssetContent_Clicked()
{
TSharedRef<SWindow> CookbookWindow = SNew(SWindow)
.Title(FText::FromString(TEXT("Cookbook Window")))
.ClientSize(FVector2D(800, 400))
.SupportsMaximize(false)
.SupportsMaximize(false);
IMainFrameModule& MainFrameModule = FModuleManager::LoadModuleChecked<IMainFrameModule>(TEXT("MainFrame"));
if (MainFrameModule.GetParentWindow().IsValid())
{
FSlateApplication::Get().AddWindowAsNativeChild(CookbookWindow, MainFrameModule.GetParentWindow().ToSharedRef());
}
else
{
FSlateApplication::Get().AddWindow(CookbookWindow);
}
}
在Editor Module startModule中
IAssetTools& AssetTools = FModuleManager::LoadModuleChecked<FAssetToolsModule>("AssetTools").Get();
auto Actions = MakeShareable(new FMyCustomAssetActions);
AssetTools.RegisterAssetTypeActions(Actions);
删除的时候 在shutdownModule中
//删除command
if (DisplayTestCommand)
{
IConsoleManager::Get().UnregisterConsoleObject(DisplayTestCommand);
DisplayTestCommand = nullptr;
}