[UE5] Gameplay Ability System (GAS)をUE5に導入する (β版記事)

Post 2022年4月10日日曜日

GameplayAbility GASでオンラインシリーズ UE5 Unreal Engine オンラインゲーム

今回はGameplay Ability System (通称GAS)をUE5に導入する方法について解説します。

もっと簡単な方法が出てくる可能性があるので、暫定とさせてください。

お約束
この記事作成にあたって使用した主なUnreal Engine バージョンUE 5.0.0

本日のゴール

Unreal Engine 5.0 にC++を使ってGASを導入する

- Input Bindを成功させる

- Gameplay Abilityを使えるように

- Gameplay Effectを使えて、AttributeがReplicateされるように


GAS 導入 作業前提編

この手順書ではC++を弄ります。

Windowsの方はVisual Studioの導入を完了しておいてください

Macの方はXcodeの導入を完了しておいてください

プロジェクトはC++向けで作成しておいてください。


GAS 導入 準備編1

手順書を作ってきましたので、使ってください。

① GameplayAbiitiesのプラグインをONにして、プロジェクトを再起動


②コンテンツツリーでC++のフォルダ → 「プロジェクト名」のフォルダを選択  ※Windowsではもう少し複雑なフォルダ構造になっていると思いますが、階層は同じです


③右クリックして新規C++クラスを作成


④全てのクラス → "Attribute"と検索 → "AttributeSet"を選択し→「次へ」


⑤適当に名前をつけてクラスを作成してください。特に思い入れがなければ「MyAttributeSet」のままでいいです。(※このファイル名にすれば、以降の作業が楽になります笑)

以降この作成したファイルは「MyAttributeSet」と呼称します。


⑥同じ手順で、次は「GameplayAbility」クラスを新規に作成します。今度は色々候補が多いので間違えないでください。特に思い入れがなければ「MyGameplayAbility」のままでいいです。以降この作成したファイルは「MyGameplayAbility」と呼称します。



⑦これで準備段階は完了



GAS 導入 C++弄る編 準備編

まず、プロジェクトのフォルダに移動します。すると「プロジェクト名.sln」「プロジェクト名.xcworkspace」のどちらかがあるはずなので、開きます。

※もしないですって人がいたら、プロジェクト.uprojectを

Windowsの人 : 右クリックして Generate Visual Studio~ を選択してください。

Macの人 : 選択状態にして、上のステータスバー?の「Finder」→「サービス」→「Generate Xcode~」を選択します


さて、そうすると、Windowsなら右 macなら左のツリーウィンドウから先程のフォルダが見えると思います。

Macの場合


まあこんなしちめんどうなことをしなくても、ファイルを右クリックして「編集」をクリックすることで勝手にアプリは起動するんですけどね。笑


GAS 導入 C++弄る編 本編

いよいよ弄ります。まずはもともとあったファイルを弄っていきます。※この講座では何も変更されていない方を前提として進めているので、すでに弄ったプロジェクトを前提にする場合は注意してください。


1「プロジェクト名.Build.cs」

下記を"PublicDependencyModuleNames.AddRange"内に追加します。

※左側の01.は気にしないでください。行の番号なので…。

  1. "GameplayAbilities", "GameplayTags", "GameplayTasks"

追加すると、多分こんな感じになると思います。

  1. PublicDependencyModuleNames.AddRange(new string[] { "GameplayAbilities", "GameplayTags", "GameplayTasks", "Core", "CoreUObject", "Engine", "InputCore" });

もうすでにプロジェクトを弄っていた方は上記は変わっていると思うので注意してください。


2「プロジェクト名.h」

下記を#include "CoreMinimal.h"の下に追加します。

  1. UENUM(BlueprintType)
  2. enum class AbilityInput : uint8
  3. {
  4.     // 0 None
  5.     None UMETA(DisplayName = "None"),
  6.     // 1 Confirm
  7.     Confirm UMETA(DisplayName = "Confirm"),
  8.     // 2 Cancel
  9.     Cancel UMETA(DisplayName = "Cancel"),
  10.     // 3 Pass Snow Ball
  11.     Pass UMETA(DisplayName = "Pass"),
  12.     // 4 Throw
  13.     Throw UMETA(DisplayName = "Throw"),
  14.     // 6 Make Snow Ball
  15.     Generate UMETA(DisplayName = "Generate"),
  16.     // 7 Map
  17.     Map UMETA(DisplayName = "Map"),
  18.     // 8 Call
  19.     Call UMETA(DisplayName = "Call"),
  20. };

これは、Gameplay Abilityとキーのバインドです。つまりなんかのキーを押すと自動的にGameplay Abilityが起動するようになります。

後ほどインプット設定するのですが、それと対応してます。


3-1「プロジェクト名Character.h」

下記の「XXXXXXXXXX」を使っているプロジェクト名にしてください。

「MyAttributeSet」を作成したAttributeSet名にしてください。

「MyGameplayAbility」を作成したGA名にしてください。


まずはヘッダーを編集します。05〜07を追加します。

  1. #include "CoreMinimal.h"
  2. #include "GameFramework/Character.h"
  3. #include "AbilitySystemInterface.h"

  4. #include "XXXXXXXXXX.h"
  5. #include "MyGameplayAbility.h"
  6. #include "MyAttributeSet.h"

  7. #include "XXXXXXXXXXCharacter.generated.h"



次に、下記画像のようにpublic : の下で  };の前に下記を追加します。

  1.     /** ability system */
  2.     UPROPERTY(VisibleAnywhere, BlueprintReadOnly, Category = Abilities, meta = (AllowPrivateAccess = "true"))
  3.         class UAbilitySystemComponent* AbilitySystem;
  4.     
  5.     UAbilitySystemComponent* GetAbilitySystemComponent() const
  6.     {
  7.         return AbilitySystem;
  8.     };
  9.     /** ability list */
  10.     UPROPERTY(EditAnywhere, BlueprintReadWrite, Category = Abilities)
  11.         TArray<TSubclassOf<class UMyGameplayAbility>> AbilityList;
  12.     
  13.     /** BeginPlay, PossessedBy override */
  14.     virtual void PossessedBy(AController* NewController) override;
  15.     
  16.     // Called when the game starts or when spawned
  17.     virtual void BeginPlay() override;
  18.     
  19.     /** AttributeSet */
  20.     UPROPERTY(VisibleAnywhere, BlueprintReadWrite, Category = Abilities)
  21.         UMyAttributeSet* AttributeSet;
  22.     
  23.     UFUNCTION(BlueprintCallable, Category = "PlayerState")
  24.     bool IsAlive() const;
  25.     
  26.     UFUNCTION(BlueprintCallable)
  27.         virtual float GetHealth() const;
  28.     UFUNCTION(BlueprintNativeEvent, Category= "GameplayAbility|My")
  29.     void Die();
  30.     virtual void Die_Implementation();
  31.     
  32.     UFUNCTION(BlueprintCallable, Category = "GameplayAbility|My")
  33.     virtual void CancelCharacterAllAbilities();
  34.     
  35.     virtual void InitializeAbility();
  36.     
  37.     UPROPERTY(BlueprintAssignable, Category = "Delegates")
  38.     FHealthChange OnHealthChange;
  39.     
  40. protected:
  41.     FGameplayTag DeadTag;
  42.     // Attribute changed callbacks
  43.     virtual void HealthAttributeUpdated(const FOnAttributeChangeData& Data);
  44.     
  45.     virtual void OnRep_PlayerState() override;
  46.     
  47.     void BindASCInput();
  48.     bool bASCInputBound;
  49.     


3-2「プロジェクト名Character.cpp」

変更点を示すのが面倒なのでフルリストを記述します。追加箇所はLine07,55,56,57,60,88,そして142行以下です。

下記の「XXXXXXXXXX」を使っているプロジェクト名にしてください。

「MyAttributeSet」を作成したAttributeSet名にしてください。

  1. // Copyright Epic Games, Inc. All Rights Reserved.
  2. #include "XXXXXXXXXXCharacter.h"
  3. #include "Camera/CameraComponent.h"
  4. #include "Components/CapsuleComponent.h"
  5. #include "Components/InputComponent.h"
  6. #include "AbilitySystemComponent.h" // 追加点
  7. #include "GameFramework/CharacterMovementComponent.h"
  8. #include "GameFramework/Controller.h"
  9. #include "GameFramework/SpringArmComponent.h"
  10. //////////////////////////////////////////////////////////////////////////
  11. // AXXXXXXXXXXCharacter
  12. AXXXXXXXXXXCharacter::AXXXXXXXXXXCharacter()
  13. {
  14.     // Set size for collision capsule
  15.     GetCapsuleComponent()->InitCapsuleSize(42.f, 96.0f);
  16.     // set our turn rate for input
  17.     TurnRateGamepad = 50.f;
  18.     // Don't rotate when the controller rotates. Let that just affect the camera.
  19.     bUseControllerRotationPitch = false;
  20.     bUseControllerRotationYaw = false;
  21.     bUseControllerRotationRoll = false;
  22.     // Configure character movement
  23.     GetCharacterMovement()->bOrientRotationToMovement = true; // Character moves in the direction of input...    
  24.     GetCharacterMovement()->RotationRate = FRotator(0.0f, 500.0f, 0.0f); // ...at this rotation rate
  25.     // Note: For faster iteration times these variables, and many more, can be tweaked in the Character Blueprint
  26.     // instead of recompiling to adjust them
  27.     GetCharacterMovement()->JumpZVelocity = 700.f;
  28.     GetCharacterMovement()->AirControl = 0.35f;
  29.     GetCharacterMovement()->MaxWalkSpeed = 500.f;
  30.     GetCharacterMovement()->MinAnalogWalkSpeed = 20.f;
  31.     GetCharacterMovement()->BrakingDecelerationWalking = 2000.f;
  32.     // Create a camera boom (pulls in towards the player if there is a collision)
  33.     CameraBoom = CreateDefaultSubobject<USpringArmComponent>(TEXT("CameraBoom"));
  34.     CameraBoom->SetupAttachment(RootComponent);
  35.     CameraBoom->TargetArmLength = 400.0f; // The camera follows at this distance behind the character    
  36.     CameraBoom->bUsePawnControlRotation = true; // Rotate the arm based on the controller
  37.     // Create a follow camera
  38.     FollowCamera = CreateDefaultSubobject<UCameraComponent>(TEXT("FollowCamera"));
  39.     FollowCamera->SetupAttachment(CameraBoom, USpringArmComponent::SocketName); // Attach the camera to the end of the boom and let the boom adjust to match the controller orientation
  40.     FollowCamera->bUsePawnControlRotation = false; // Camera does not rotate relative to arm
  41.     // Note: The skeletal mesh and anim blueprint references on the Mesh component (inherited from Character)
  42.     // are set in the derived blueprint asset named ThirdPersonCharacter (to avoid direct content references in C++)
  43.     
  44.     // ability system component
  45.     AbilitySystem = CreateDefaultSubobject<UAbilitySystemComponent>(TEXT("AbilitySystem")); // 追加点
  46.     AbilitySystem->SetIsReplicated(true); // 追加点
  47.     AbilitySystem->SetReplicationMode(EGameplayEffectReplicationMode::Mixed); // 追加点
  48.     
  49.     // create the attribute set
  50.     AttributeSet = CreateDefaultSubobject<UMyAttributeSet>(TEXT("AttributeSet")); // 追加点
  51. }
  52. //////////////////////////////////////////////////////////////////////////
  53. // Input
  54. void AXXXXXXXXXXCharacter::SetupPlayerInputComponent(class UInputComponent* PlayerInputComponent)
  55. {
  56.     // Set up gameplay key bindings
  57.     check(PlayerInputComponent);
  58.     PlayerInputComponent->BindAction("Jump", IE_Pressed, this, &ACharacter::Jump);
  59.     PlayerInputComponent->BindAction("Jump", IE_Released, this, &ACharacter::StopJumping);
  60.     PlayerInputComponent->BindAxis("Move Forward / Backward", this, &AXXXXXXXXXXCharacter::MoveForward);
  61.     PlayerInputComponent->BindAxis("Move Right / Left", this, &AXXXXXXXXXXCharacter::MoveRight);
  62.     // We have 2 versions of the rotation bindings to handle different kinds of devices differently
  63.     // "turn" handles devices that provide an absolute delta, such as a mouse.
  64.     // "turnrate" is for devices that we choose to treat as a rate of change, such as an analog joystick
  65.     PlayerInputComponent->BindAxis("Turn Right / Left Mouse", this, &APawn::AddControllerYawInput);
  66.     PlayerInputComponent->BindAxis("Turn Right / Left Gamepad", this, &AXXXXXXXXXXCharacter::TurnAtRate);
  67.     PlayerInputComponent->BindAxis("Look Up / Down Mouse", this, &APawn::AddControllerPitchInput);
  68.     PlayerInputComponent->BindAxis("Look Up / Down Gamepad", this, &AXXXXXXXXXXCharacter::LookUpAtRate);
  69.     // handle touch devices
  70.     PlayerInputComponent->BindTouch(IE_Pressed, this, &AXXXXXXXXXXCharacter::TouchStarted);
  71.     PlayerInputComponent->BindTouch(IE_Released, this, &AXXXXXXXXXXCharacter::TouchStopped);
  72.     
  73.     BindASCInput(); // 追加点
  74. }
  75. void AXXXXXXXXXXCharacter::TouchStarted(ETouchIndex::Type FingerIndex, FVector Location)
  76. {
  77.     Jump();
  78. }
  79. void AXXXXXXXXXXCharacter::TouchStopped(ETouchIndex::Type FingerIndex, FVector Location)
  80. {
  81.     StopJumping();
  82. }
  83. void AXXXXXXXXXXCharacter::TurnAtRate(float Rate)
  84. {
  85.     // calculate delta for this frame from the rate information
  86.     AddControllerYawInput(Rate * TurnRateGamepad * GetWorld()->GetDeltaSeconds());
  87. }
  88. void AXXXXXXXXXXCharacter::LookUpAtRate(float Rate)
  89. {
  90.     // calculate delta for this frame from the rate information
  91.     AddControllerPitchInput(Rate * TurnRateGamepad * GetWorld()->GetDeltaSeconds());
  92. }
  93. void AXXXXXXXXXXCharacter::MoveForward(float Value)
  94. {
  95.     if ((Controller != nullptr) && (Value != 0.0f))
  96.     {
  97.         // find out which way is forward
  98.         const FRotator Rotation = Controller->GetControlRotation();
  99.         const FRotator YawRotation(0, Rotation.Yaw, 0);
  100.         // get forward vector
  101.         const FVector Direction = FRotationMatrix(YawRotation).GetUnitAxis(EAxis::X);
  102.         AddMovementInput(Direction, Value);
  103.     }
  104. }
  105. void AXXXXXXXXXXCharacter::MoveRight(float Value)
  106. {
  107.     if ( (Controller != nullptr) && (Value != 0.0f) )
  108.     {
  109.         // find out which way is right
  110.         const FRotator Rotation = Controller->GetControlRotation();
  111.         const FRotator YawRotation(0, Rotation.Yaw, 0);
  112.     
  113.         // get right vector
  114.         const FVector Direction = FRotationMatrix(YawRotation).GetUnitAxis(EAxis::Y);
  115.         // add movement in that direction
  116.         AddMovementInput(Direction, Value);
  117.     }
  118. }
  119. void AXXXXXXXXXXCharacter::BeginPlay()
  120. {
  121.     Super::BeginPlay();
  122.     if (AbilitySystem)
  123.     {
  124.         int32 inputID(0);
  125.         if (HasAuthority() && AbilityList.Num() > 0)
  126.         {
  127.             for (auto Ability : AbilityList)
  128.             {
  129.                 if (Ability)
  130.                 {
  131.                     if (static_cast<int32>(Ability.GetDefaultObject()->AbilityInputID) != 0)
  132.                     {
  133.                         AbilitySystem->GiveAbility(FGameplayAbilitySpec(Ability.GetDefaultObject(), 1, static_cast<int32>(Ability.GetDefaultObject()->AbilityInputID), this));
  134.                     }
  135.                     else
  136.                     {
  137.                         AbilitySystem->GiveAbility(FGameplayAbilitySpec(Ability.GetDefaultObject(), 1, -1, this));
  138.                     }
  139.                 }
  140.             }
  141.         }
  142.         AbilitySystem->InitAbilityActorInfo(this, this);
  143.         
  144.         AbilitySystem->GetGameplayAttributeValueChangeDelegate(UMyAttributeSet::GetHealthAttribute()).AddUObject(this, &AXXXXXXXXXXCharacter::HealthAttributeUpdated);
  145.         
  146.     }
  147.     
  148.     
  149. }
  150. void AXXXXXXXXXXCharacter::InitializeAbility()
  151. {
  152.     
  153.     if (AbilitySystem)
  154.     {
  155.         int32 inputID(0);
  156.         if (HasAuthority() && AbilityList.Num() > 0)
  157.         {
  158.             for (auto Ability : AbilityList) {
  159.                 if (Ability)
  160.                 {
  161.                     AbilitySystem->GiveAbility(FGameplayAbilitySpec(Ability.GetDefaultObject(), 1, inputID++));
  162.                 }
  163.             }
  164.             
  165.             
  166.         }
  167.         AbilitySystem->InitAbilityActorInfo(this, this);
  168.         
  169.         
  170.     }
  171.     
  172. }
  173. void AXXXXXXXXXXCharacter::PossessedBy(AController* NewController)
  174. {
  175.     Super::PossessedBy(NewController);
  176.         if (AbilitySystem)
  177.         {
  178.             AbilitySystem->InitAbilityActorInfo(this, this);
  179.         }
  180.         // ASC MixedMode レプリケーションでは、ASC Owner の所有者に Controller を要求します。
  181.         SetOwner(NewController);
  182.     
  183. }
  184. float AXXXXXXXXXXCharacter::GetHealth() const {
  185.     return AttributeSet->GetHealth();
  186. }
  187. bool AXXXXXXXXXXCharacter::IsAlive() const
  188. {
  189.     return GetHealth() > 0.0f;
  190. }
  191. void AXXXXXXXXXXCharacter::Die_Implementation()
  192. {
  193. }
  194. void AXXXXXXXXXXCharacter::HealthAttributeUpdated(const FOnAttributeChangeData& Data)
  195. {
  196.     OnHealthChange.Broadcast(Data.NewValue);
  197.     
  198.     if (!IsAlive() && !AbilitySystem->HasMatchingGameplayTag(DeadTag))
  199.     {
  200.             AbilitySystem->AddLooseGameplayTag(DeadTag);
  201.             Die();
  202.     }
  203.     
  204. }
  205. void AXXXXXXXXXXCharacter::CancelCharacterAllAbilities()
  206. {
  207.     if (GetLocalRole() != ROLE_Authority || !AbilitySystem)
  208.     {
  209.         return;
  210.     }
  211.     AbilitySystem->CancelAllAbilities();
  212. }
  213. void AXXXXXXXXXXCharacter::OnRep_PlayerState()
  214. {
  215.     Super::OnRep_PlayerState();
  216.     AbilitySystem->InitAbilityActorInfo(this, this);
  217.     BindASCInput();
  218. }
  219. void AXXXXXXXXXXCharacter::BindASCInput()
  220. {
  221.     if (!bASCInputBound && IsValid(AbilitySystem) && IsValid(InputComponent))
  222.     {
  223.         AbilitySystem->BindAbilityActivationToInputComponent(InputComponent, FGameplayAbilityInputBinds(FString("ConfirmTarget"),
  224.             FString("CancelTarget"), FString("AbilityInput"), static_cast<int32>(AbilityInput::Confirm), static_cast<int32>(AbilityInput::Cancel)));
  225.         bASCInputBound = true;
  226.     }
  227. }


4-1「MyAttributeSet.h」

下記の「XXXXXXXXXX」を使っているプロジェクト名にしてください。

「MyAttributeSet」を作成したAttributeSet名にしてください。

  1. // Fill out your copyright notice in the Description page of Project Settings.
  2. #pragma once
  3. #include "CoreMinimal.h"
  4. #include "AttributeSet.h"
  5. #include "AbilitySystemComponent.h"
  6. #include "MyAttributeSet.generated.h"
  7. // AttributeSetを引っ張ってこれるための準備
  8. #define ATTRIBUTE_ACCESSORS(ClassName, PropertyName) \
  9.     GAMEPLAYATTRIBUTE_PROPERTY_GETTER(ClassName, PropertyName) \
  10.     GAMEPLAYATTRIBUTE_VALUE_GETTER(PropertyName) \
  11.     GAMEPLAYATTRIBUTE_VALUE_SETTER(PropertyName) \
  12.     GAMEPLAYATTRIBUTE_VALUE_INITTER(PropertyName)
  13. UCLASS()
  14. class XXXXXXXXXX_API UMyAttributeSet : public UAttributeSet
  15. {
  16.     GENERATED_BODY()
  17.     
  18. public:
  19.     UMyAttributeSet();
  20.     
  21.     virtual void GetLifetimeReplicatedProps(TArray<FLifetimeProperty>& OutLifetimeProps) const override;
  22.     
  23.     //
  24.     UPROPERTY(Category = "MyAttributeSet", EditAnywhere, BlueprintReadWrite, ReplicatedUsing = OnRep_Health)
  25.         FGameplayAttributeData Health;
  26.     ATTRIBUTE_ACCESSORS(UMyAttributeSet, Health);
  27.     FGameplayAttribute HealthAttribute();
  28.     
  29.     // Yukidama
  30.     UPROPERTY(BlueprintReadWrite, Category = "MyAttributeSet", ReplicatedUsing = OnRep_Yukidama)
  31.         FGameplayAttributeData Yukidama;
  32.     ATTRIBUTE_ACCESSORS(UMyAttributeSet, Yukidama);
  33.     FGameplayAttribute YukidamaAttribute();
  34.     
  35.     // YukidamaMax
  36.     UPROPERTY(BlueprintReadWrite, Category = "MyAttributeSet", EditAnywhere)
  37.         FGameplayAttributeData YukidamaMax;
  38.     ATTRIBUTE_ACCESSORS(UMyAttributeSet, YukidamaMax);
  39.     FGameplayAttribute YukidamaMaxAttribute();
  40.     
  41.     //
  42.     virtual void PostGameplayEffectExecute(const FGameplayEffectModCallbackData& Data) override;
  43.     
  44. protected:
  45.     
  46.     UFUNCTION()
  47.     virtual void OnRep_Yukidama(const FGameplayAttributeData& OldYukidama);
  48.     
  49.     UFUNCTION()
  50.     virtual void OnRep_Health(const FGameplayAttributeData& OldHealth);
  51.     
  52. };

4-2「MyAttributeSet.cpp」

下記の「XXXXXXXXXX」を使っているプロジェクト名にしてください。

「MyAttributeSet」を作成したAttributeSet名にしてください。

  1. // Fill out your copyright notice in the Description page of Project Settings.
  2. #include "MyAttributeSet.h"
  3. #include "GameplayEffect.h"
  4. #include "GameplayEffectExtension.h" // ForPostGameplayEffectExecute
  5. #include "XXXXXXXXXXCharacter.h" // Character
  6. #include "Net/UnrealNetwork.h"
  7. UMyAttributeSet::UMyAttributeSet()
  8.     : Health(1.f)
  9.     , Yukidama(5.f)
  10.     , YukidamaMax(5.f)
  11. {
  12. }
  13. // Health
  14. FGameplayAttribute UMyAttributeSet::HealthAttribute()
  15. {
  16.     static FProperty* Property = FindFieldChecked<FProperty>(UMyAttributeSet::StaticClass(), GET_MEMBER_NAME_CHECKED(UMyAttributeSet, Health));
  17.     return FGameplayAttribute(Property);
  18. }
  19. FGameplayAttribute UMyAttributeSet::YukidamaAttribute()
  20. {
  21.     static FProperty* Property = FindFieldChecked<FProperty>(UMyAttributeSet::StaticClass(), GET_MEMBER_NAME_CHECKED(UMyAttributeSet, Yukidama));
  22.     return FGameplayAttribute(Property);
  23. }
  24. FGameplayAttribute UMyAttributeSet::YukidamaMaxAttribute()
  25. {
  26.     static FProperty* Property = FindFieldChecked<FProperty>(UMyAttributeSet::StaticClass(), GET_MEMBER_NAME_CHECKED(UMyAttributeSet, YukidamaMax));
  27.     return FGameplayAttribute(Property);
  28. }
  29. void UMyAttributeSet::PostGameplayEffectExecute(const FGameplayEffectModCallbackData& Data)
  30. {
  31.     Super::PostGameplayEffectExecute(Data);
  32.     // éÛÇØéÊÇ¡ÇΩÉfÅ[É^Ç©ÇÁäeéÌèÓïÒÇ?éÊìæ
  33.     FGameplayEffectContextHandle Context = Data.EffectSpec.GetContext();
  34.     UAbilitySystemComponent* Source = Context.GetOriginalInstigatorAbilitySystemComponent();
  35.     const FGameplayTagContainer& SourceTags = *Data.EffectSpec.CapturedSourceTags.GetAggregatedTags();
  36.     // GameplayEffectÇ…ÇÊÇËéwíËÇ?ÇÍÇΩÉAÉgÉäÉrÉÖÅ[ÉgïœâªílÇ?åvéZ
  37.     float DeltaValue = 0;
  38.     if (Data.EvaluatedData.ModifierOp == EGameplayModOp::Type::Additive)
  39.     {
  40.         // If this was additive, store the raw delta value to be passed along later
  41.         DeltaValue = Data.EvaluatedData.Magnitude;
  42.     }
  43.     AActor* TargetActor = nullptr;
  44.     AController* TargetController = nullptr;
  45.     AXXXXXXXXXXCharacter* TargetCharacter = nullptr;
  46.     if (Data.Target.AbilityActorInfo.IsValid() && Data.Target.AbilityActorInfo->AvatarActor.IsValid())
  47.     {
  48.         TargetActor = Data.Target.AbilityActorInfo->AvatarActor.Get();
  49.         TargetController = Data.Target.AbilityActorInfo->PlayerController.Get();
  50.         TargetCharacter = Cast<AXXXXXXXXXXCharacter>(TargetActor);
  51.     }
  52.     
  53.     if (GetYukidamaAttribute() == GetYukidamaMaxAttribute())
  54.     {
  55.         // Syori
  56.     }
  57.     
  58. }
  59. void UMyAttributeSet::OnRep_Yukidama(const FGameplayAttributeData& OldYukidama)
  60. {
  61.     GAMEPLAYATTRIBUTE_REPNOTIFY(UMyAttributeSet, Yukidama, OldYukidama);
  62. }
  63. void UMyAttributeSet::OnRep_Health(const FGameplayAttributeData& OldHealth)
  64. {
  65.     GAMEPLAYATTRIBUTE_REPNOTIFY(UMyAttributeSet, Health, OldHealth);
  66.     
  67. }
  68. void UMyAttributeSet::GetLifetimeReplicatedProps(TArray<FLifetimeProperty>& OutLifetimeProps) const
  69. {
  70.     Super::GetLifetimeReplicatedProps(OutLifetimeProps);
  71.     DOREPLIFETIME_CONDITION_NOTIFY(UMyAttributeSet, Yukidama, COND_None, REPNOTIFY_Always);
  72.     
  73.     DOREPLIFETIME_CONDITION_NOTIFY(UMyAttributeSet, Health, COND_None, REPNOTIFY_Always);
  74. }

5-1「MyGameAbility.h」

下記の「XXXXXXXXXX」を使っているプロジェクト名にしてください。

「MyGameplayAbility」を作成したGA名にしてください。

  1. // Fill out your copyright notice in the Description page of Project Settings.
  2. #pragma once
  3. #include "CoreMinimal.h"
  4. #include "XXXXXXXXXX.h"
  5. #include "Abilities/GameplayAbility.h"
  6. #include "MyGameplayAbility.generated.h"
  7. UCLASS()
  8. class XXXXXXXXXX_API UMyGameplayAbility : public UGameplayAbility
  9. {
  10.     GENERATED_BODY()
  11.     
  12. public:
  13.     UMyGameplayAbility();
  14.     
  15.     // Abilities with this set will automatically activate when the input is pressed
  16.     UPROPERTY(BlueprintReadOnly, EditAnywhere, Category = "Ability")
  17.     AbilityInput AbilityInputID = AbilityInput::None;
  18.     // Value to associate an ability with an slot without tying it to an automatically activated input.
  19.     // Passive abilities won't be tied to an input so we need a way to generically associate abilities with slots.
  20.     UPROPERTY(BlueprintReadOnly, EditAnywhere, Category = "Ability")
  21.     AbilityInput AbilityID = AbilityInput::None;
  22. };


5-2「MyGameAbility.cpp」

「MyGameplayAbility」を作成したGA名にしてください。

  1. // Fill out your copyright notice in the Description page of Project Settings.
  2. #include "MyGameplayAbility.h"
  3. UMyGameplayAbility::UMyGameplayAbility()
  4. {
  5. }


C++をコンパイルする

C++のコンパイルは、BPのコンパイルボタンとは別にあります。
これです。これを押して暫く待つと、コンパイルが終わります。

多分今回の実装で、エラーが起こることはないと思うのですが、エラーが起こったらエラーの内容をコメントで書いてください…。

GASを作る

さて、これで準備は整いました。先程作った「MyGameplayAbility」を右クリックして、これを継承したBPを作成しましょう。作成したら早速開いてください。
Ability Input ID / Ability ID

これ(Ability Input ID / Ability ID)を設定していきます。

詳しくはらくするさんのページ(上)を参照してほしいのですが、
このInput IDを設定することで、最終的にインプットとバインドすることができ、インプットがあれば自動的にGASを起動できます。


われわれは2の項でこのような設定をしました。

---

2「プロジェクト名.h」

下記を#include "CoreMinimal.h"の下に追加します。

  1. UENUM(BlueprintType)
  2. enum class AbilityInput : uint8
  3. {
  4.     // 0 None
  5.     None UMETA(DisplayName = "None"),
  6.     // 1 Confirm
  7.     Confirm UMETA(DisplayName = "Confirm"),
  8.     // 2 Cancel
  9.     Cancel UMETA(DisplayName = "Cancel"),
  10.     // 3 Pass Snow Ball
  11.     Pass UMETA(DisplayName = "Pass"),
  12.     // 4 Throw
  13.     Throw UMETA(DisplayName = "Throw"),
  14.     // 6 Make Snow Ball
  15.     Generate UMETA(DisplayName = "Generate"),
  16.     // 7 Map
  17.     Map UMETA(DisplayName = "Map"),
  18.     // 8 Call
  19.     Call UMETA(DisplayName = "Call"),
  20. };

----

Ability Input ID / Ability IDをクリックすると、これのリストが出てくるはずです。

今回作った、GA(Gameplay Ability)はPassに相当するGAだったので、Passを選択しました。
例えば敵キャラしか適用しないGAなんかもあるでしょう。その場合は操作キャラが使うのはおかしいので、そういったGAでは「None」を選ぶことになります。

ちなみにAbility Input ID/Ability IDの違いは、正直ないです。(おい)
本当はAbility IDの方は、「Inputにはバインドしたくないけど、Input Keyとして扱いたい」場合に設定するんですけどね。それの内容をプログラミングしてないんで動きません。(おい)


Inputの設定

プロジェクト設定 → 「インプット」で設定できます。
アクションマッピングで設定しますが、このとき、先ほどのリストと同じ名前で設定するようにお願いします。でないと起動しませんので…


忘れずに…

キャラクターのBPで、Ability ListにGAを登録してくださいね。



これで、C++の設定を無視してGASシリーズのPart1〜Part6を見ることができます。