使用UE:Ver5.1.0
使用VS:VS2019
5.1.0からEnhanced Inputが推奨されるInput方法となったので調べてみました。
data:image/s3,"s3://crabby-images/724c0/724c02c59815ace365b9b98d1d15bf40ec9b35c9" alt=""
Enhanced Inputの基本
必要になるのは、Input Action(IA)とInput Mapping Context(IMC)の2つ。
data:image/s3,"s3://crabby-images/7263e/7263eefb3ba7e2246d618a063dd2d6c094c45055" alt=""
data:image/s3,"s3://crabby-images/24cb7/24cb7dffb4a2e870d73dcd0dce8d8da9648b9014" alt=""
data:image/s3,"s3://crabby-images/c77bc/c77bc466e1342f8db3be56e718c001e99b73cc57" alt=""
Input Action(IA)
動作それぞれに対してつくるもの。
動作についての詳細を記述できる。
詳しくは後で説明する。
data:image/s3,"s3://crabby-images/768c8/768c8003642b66259196487c6820277ef7953591" alt=""
ThirdPersonのテンプレートでは、次の3つのInput Actionが用意されている。
data:image/s3,"s3://crabby-images/810d2/810d29557a11fabf9b9dfc0bc5b512b13cdcab1f" alt=""
Input Mapping Context(IMC)
Input Actionを設定して、キーを結び付けるところ。
data:image/s3,"s3://crabby-images/a22ea/a22eafcbe3c1486007d3e7ffc00fdeb679013798" alt=""
下はThirdPersonテンプレートのIMC。
data:image/s3,"s3://crabby-images/73451/73451ebcb973efcbf0f3cd945080df8e6b3b732e" alt=""
IMCの着脱
Add Mapping Contextにより、IMCが使用可能になる。
Remove Mapping Contextにより、IMCが使用不可能になる。
IMCが使用可能なときだけ、その中に設定されているIAが使用できる。
下の例だと、右のマウスを押しているときIMCが使用可能。
右のマウスを離すと使用不可能になる。
data:image/s3,"s3://crabby-images/30099/30099e00e45b0391ea9fd52302e291703c621000" alt=""
詳細設定
Input Actionにおいて設定できる、Value Type、Triggers、Modifiersの3つについて考える。
(TriggersとModifiersについてはIMCでも設定できる。)
data:image/s3,"s3://crabby-images/faa11/faa11f560d496d295a8ebccb6b73e3bb5a042c99" alt=""
Value Type
Input ActionのValue Typeは、以下の4種類の変数型から選べる。
data:image/s3,"s3://crabby-images/94592/9459254187931028b9d8791416b914af698f10a4" alt=""
Value Typeで決めた変数の型が、イベントノードのAction Valueの変数の型になる。
実行ピンの数が以前より増えて戸惑う方もいると思う。
それはTriggersの設定と関係があるので、後に説明する。
data:image/s3,"s3://crabby-images/4b623/4b6235b2b6915a1c5fde8243effd78da4e73a363" alt=""
ThirdPersonのテンプレートでValue Typeは、
IA_Jump:bool
IA_Move:Axis2D
IA_Look:Axis2D
となっている。
Jumpは飛ぶか飛ばないか2択なのでbool値。
MoveはForwardとRightの移動のためにAxis2D。
Lookは視点変更であり、YawとPitchの回転のためAxis2D。
Triggers
次の10個から選択できる。
data:image/s3,"s3://crabby-images/58ef4/58ef48e362f45c5834dbdfe953af992ebc8f9f64" alt=""
どのEventが実行されるかは、TriggerState(TState)というものの変化によって決まる。
data:image/s3,"s3://crabby-images/8e266/8e266fc93f846e7e65403c516fce928c1a4d7cdf" alt=""
TStateにはNone、Ongoing、Triggeredの3種類がある。
それらの変化の仕方で実行されるEventが決まる。
👇EnhancedPlayerInput.cppより抜粋したもの
// LastTState NewTState Event
// None -> Ongoing = Started
// None -> Triggered = Started + Triggered
// Ongoing -> None = Canceled
// Ongoing -> Ongoing = Ongoing
// Ongoing -> Triggered = Triggered
// Triggered -> Triggered = Triggered
// Triggered -> Ongoing = Ongoing
// Triggered -> None = Completed
TStateの変化の仕方については、実際試して理解した方が早そう。
上記のブログで行われていた可視化の方法を、自分で試してみて分かったことをまとめた。
TStateの変化の仕方はTriggersの設定によって異なる。
Triggersの設定ごとのTStateとEventについてまとめる。
括弧の中身がEventの種類で、その左に書いてあるのがTStateの変化である。
None
たぶんDownと同じ。DownがDefaultなのだと思う。
Down
キーを押す:None→Triggered(Started + Triggered)
押している間:Triggered→Triggered(Triggered)
キーを離す:Triggered→None(Completed)
Hold
キーを押す:None→Ongoing(Started)
押している間:Ongoing→Ongoing(Ongoing)
時間経過(Hold Time Threshold):Ongoing→Triggered(Triggered)
その後押している間:Triggered→Triggered(Triggered)
Ongoing時に離す:Ongoing→None(Canceled)
Triggered時に離す:Triggered→None(Completed)
Hold And Release
キーを押す:None→Ongoing(Started)
押している間:Ongoing→Ongoing(Ongoing)
離す(Hold Time Threshold未満):Ongoing→None(Canceled)
離す(Hold Time Threshold以上):Ongoing→Triggered→None(Triggered + Completed)
Pressed
キーを押す:None→Triggered→None(Started + Triggered + Completed)
キーを押し続ける:None→None(No Event)
キーを離す:None→None(No Event)
Pulse
キーを押す:None→Triggered→Ongoing(Started + Triggered + Ongoing)
キーを押し続ける:Ongoing→Ongoing(Ongoing)
時間経過ごと(Interval):Ongoing→Triggered→Ongoing(Triggered + Ongoing)
キーを離す:Ongoing→None(Canceled)
Released
キーを押す:None→Ongoing(Started)
キーを押し続ける:Ongoing→Ongoing(Ongoing)
キーを離す:Ongoing→Triggered→None(Triggered + Completed)
Tap
キーを押す:None→Ongoing(Started)
キーを押し続ける:Ongoing→Ongoing(Ongoing)
離す(Tap Release Time Threshold未満):Ongoing→Triggered→None(Triggered + Completed)
時間経過(Tap Release Time Threshold以上):Ongoing→None(Canceled)
Tap Release Time Threshold以上押し続ける:None→None(No Event)
離す(Tap Release Time Threshold以上):None→None(No Event)
その他のTrigger設定
難しそうなので保留。
Chorded Action
別のInput Actionをトリガーとするらしい。
Combo
良く分からない。
👇Comboで試したら、breakpointで止められた。
data:image/s3,"s3://crabby-images/469d9/469d9ca28043f64f3bc447877fe1c424942696ac" alt=""
Modifiers
まずは、ThirdPersonテンプレートのIA_MoveのWSADに使われている2種類のModifierについて考える。
data:image/s3,"s3://crabby-images/c4f34/c4f346cbd6985aac2d88386855249699420e8595" alt=""
キーの入力(1次元の情報入力)に関して言えば、
値を入力する軸を決めるModifierだと理解して良いと思う。
OrderがYXZの場合、Y軸方向に入力。
ZYXの場合、Z軸に入力。
このModifierを使わない場合XYZ。XYZはX軸に入力。
data:image/s3,"s3://crabby-images/9cc92/9cc924cdc6013ac23237d14f294e2836fd19ba7d" alt=""
IA_Moveをみると、
W(前)とS(後ろ)には、このModifierついている。
詳細を見てみると、OrderはYXZ(Y軸)になっている。
IA_MoveのValue TypeはAxis2Dであり、
WとSの入力は、Axis2DのY軸の値を決めていることが分かる。
data:image/s3,"s3://crabby-images/60654/60654d69fda90a71b11a00581a0e8b3fb8a75f42" alt=""
A(左)とD(右)には、このModifierはついていない。
これはOrderがXYZ(X軸)であることを示している。
つまりAとDの入力は、Axis2DのX軸の値を決めていることが分かる。
推測
WASDなどのキーは1次元の情報しか入力できないので、1番初めのOrderのみに意味がある。
例えばYXZならば、Yのみに意味がある。
GamepadのThumstickのように2次元以上の情報を入力できるようになると、順番に意味が出てくるのだと思う。
XYZ(X軸)、YXZ(Y軸)、ZYX(Z軸)のどれかを使うのが無難なのだと思う。
cyclicに並んでいないのはUEが左手系なのに由来すると思うが、良く分からない。
Swizzleの意味
To permute bits, or elements of a vector.
ビットや、ベクターの要素の順序を変えること。
(ここではベクター要素の順序を変える意味で使っている)
符号(正負)を反転させるModifier。
IA_Moveを見てみる。
W(前)にNegateはついていないが、S(後ろ)にはついている。
OrderがYXZなので、WもSもY軸の値を入力する。
Negateの有無と、WSのキー入力が正の値しかとらないことより、
WはAxis2DのY軸に正の値を入力し、
SはAxis2DのY軸に負の値を入力することが分かる。
data:image/s3,"s3://crabby-images/e5862/e5862adabf0bfb9960c87ccfad231625c2293660" alt=""
AとDについても同様に考えることで、
DはAxis2DのX軸に正の値を入力し、
AはAxis2DのX軸に負の値を入力することが分かる。
data:image/s3,"s3://crabby-images/cf235/cf2358d9c5dab662d4d8d97e35975fd642c46cd2" alt=""
つぎに、同じIA_Moveに対するThumstickの入力について考える。
data:image/s3,"s3://crabby-images/4852e/4852e0aabbd5e0631f6b1833f66cd7fedf68eda3" alt=""
Thumstickの場合、これだけでAxis2Dの情報を入力できる。
Thumstickを倒す向きと入力値の関係はおそらく、
右:Axis2DのX軸に正の値を入力
左:Axis2DのX軸に負の値を入力
上:Axis2DのY軸に正の値を入力
下:Axis2DのY軸に負の値を入力
Swizzle Input Axis ValuesのModifierのOrderで順番を変えることはできそう。
推測した効果
入力値をLower ThresholdとUpper Thresholdでクランプした後に、
(入力値 – Lower) ÷ (Upper – Lower)を計算したもの。
LowerとUpperの範囲外なら検知しない。(Type:Radial)
この推測のための検証は後の見出しで行っている。
IA_Move
Thumstickによる入力値の絶対値は、おそらく0から1の間の値をとる。
その大きさはThumstickの傾きに比例して大きくなる。
そのため傾けたつもりが無くても、微小に傾いていて微小な値が入力されてしまう。
そのような意図しない入力を避けるために使われているのだと思われる。
👇公式ドキュメント
実装してみる
BlueprintでMobileのジェスチャーによる入力を実装してみる。
IMC_GestureというInput Mapping Contextを作成した。
中身はIA_Pinch、IA_Rotate、IA_Flickの3つ。(IA_Flickは実装失敗)
下準備
data:image/s3,"s3://crabby-images/59e6c/59e6cd57a47db4b6ced3bd0c98c6b0a7759e189d" alt=""
Project Settings
Enable Gesture Recognizer : true
CharacterなどへのキャストはReferenceが増えないので割と使う。
元から用意してあるクラスだからか、C++のクラスだからかは調べていない。
data:image/s3,"s3://crabby-images/5720c/5720cbef4ad6a588695d11a68b1eb984170f5359" alt=""
IA_Pinch
data:image/s3,"s3://crabby-images/b1e54/b1e54ea87e98e977345b0e15d314d84d8a64e782" alt=""
Modifierを使っていない場合、はじめに2本の指をおいた間隔でAction Value(Float)の値は1になる。
はじめの間隔を1として、指の間隔を広げると1より大きな値に、狭めると1より小さな値になる。
最大値は2を超えるが、最小値はおそらく0。
おそらく、はじめにおいた2本の指の間隔を1としたときの比を出力している。
Pinchの入力において、Dead ZoneのModifierの効果を少し調べてみた。
Type:Radial
case:1
Lower Threshold:0
Upper Threshold:2
初期間隔:0.5
最小値:0
最大値:1
case:2
Lower Threshold:0
Upper Threshold:4
初期間隔:0.25
最小値:0
最大値:1
case:3
Lower Threshold:1
Upper Threshold:5
初期間隔:検知されない
最小値:0
最大値:1
case:4
Lower Threshold:0.5
Upper Threshold:2.5
初期間隔:0.25
最小値:0(初期間隔の半分より狭くすると検知しなくなる)
最大値:1
case:5
Lower Threshold:0.8
Upper Threshold:1.3
初期間隔:0.4
最小値:0
最大値:1
推測した効果
入力値をLower ThresholdとUpper Thresholdでクランプした後に、
(入力値 – Lower) ÷ (Upper – Lower)を計算したもの。
LowerとUpperの範囲外なら検知しない。
IA_Rotate
Last Angleの初期値は0。
data:image/s3,"s3://crabby-images/f2ccd/f2ccd9715b203dd696096ff35cf4814941639f3c" alt=""
右回りで正の値を出力。左回りで負の値を出力。
90°の回転で90を出力。
UMGの要素を回転させるとき、
アンカーの種類によっては回転によりアンカーの基準が変わってしまうためか、回転の中心が期待した位置から変化してしまう。
IA_Flick
Action ValueをAxis2DにしてみたがXの値しかとれなかったので、今は実装をあきらめる。
下のブログが分かりやすそうだったので、参考させていただきタッチ系関数でいずれ実装したい。
その他の学び
5.1C++
5.1でC++のプロジェクトを作ろうとしたら、下記の表示がでた。
data:image/s3,"s3://crabby-images/2ef3f/2ef3fd949b4910afd58ae21dae8702cbc0d70930" alt=""
下のサイト(上記表示に書いてあるURL)に行くと、自動的にexeファイルがダウンロードされるので実行。
data:image/s3,"s3://crabby-images/13b5d/13b5d4f48921f6e4410147c4bf79d8d47db794fc" alt=""
6.0.11が追加された。この状態ならばC++プロジェクトを作ることができた。
data:image/s3,"s3://crabby-images/c1c82/c1c82689695102f9bdbf0f5ea60e9b5977403088" alt=""
data:image/s3,"s3://crabby-images/794af/794aface925b616cab6a68361cd46bfb57db3780" alt=""
下の警告への対応。
install
data:image/s3,"s3://crabby-images/dc89e/dc89eb2edce7cfc13b91d90bc2a8f46cc146dbe5" alt=""
インストール
data:image/s3,"s3://crabby-images/04572/045728e9016c8dcdc08cad502eee9b51189e5712" alt=""
閉じる。特に問題は起こらなかった。
data:image/s3,"s3://crabby-images/4c769/4c769da6213eb307f15eb85299dd291f8579d320" alt=""
起きたエラー
UEのVerを5.1にしてC++を使用した場合下記のエラーがでてAndroidで実機プレイできなかった。
(C++を使わない場合は実機プレイできた。)
LogPlayLevel: Error: UAT: ld.lld: error: undefined symbol: __aarch64_swp8_acq_rel
解決方法
NDKのVerが古いのが原因。
Android Studio4.0をDefaultでインストールした状態から、NDKのVerを書き換える方法を下に示す。
Android StudioのVerが4.0でない人はこちらからinstall。
まず、任意のプロジェクトをAndroid Studioで開く。
上のメニューバーからTools>SDK Managerを選択する。
data:image/s3,"s3://crabby-images/4daea/4daeaca0f8987c663be898c2ff6207c01657d800" alt=""
出てきたウィンドウの中で、SDKToolsのタブをクリック。
data:image/s3,"s3://crabby-images/0fca9/0fca9ef1bea1575cc7646eaa7b5625016196a391" alt=""
その後、Show Packge Detailsのチェックボックスにチェックを入れる。
data:image/s3,"s3://crabby-images/155a8/155a8420e9120688f441e3baf7b762e943ed97ea" alt=""
25.1.8937393のVerにチェック。
以前のVerのチェックを外すことでuninstallできる。(任意)
data:image/s3,"s3://crabby-images/67c3f/67c3ff5ef390697562ffdf93127ea0f63feb9a0f" alt=""
OKをクリック。
data:image/s3,"s3://crabby-images/c389f/c389f5a1790eb85907072fd08b62872a45ed6886" alt=""
Finishをクリック。
data:image/s3,"s3://crabby-images/cd19a/cd19a502f4935ecd36e8f8b408b5019d5130c3eb" alt=""
Project Settingsで忘れずに設定。
data:image/s3,"s3://crabby-images/95817/95817181afac79018614e4541b589bf556e49b9e" alt=""
参考
data:image/s3,"s3://crabby-images/23cbe/23cbed41f0e0deca8496c455a9d9277082a120ac" alt=""
data:image/s3,"s3://crabby-images/4817c/4817ccadc19f3e42731cf222048d66e4a0f00b8e" alt=""
data:image/s3,"s3://crabby-images/4817c/4817ccadc19f3e42731cf222048d66e4a0f00b8e" alt=""
Virtual Joystick
Project Settings>Engine>Input>Mobile
で設定できる。
data:image/s3,"s3://crabby-images/b5745/b574579703c834da9bda3e88be3a2fd7adcd85cc" alt=""
DefaultのInterface(Engine Content)をコピーして、自分用のInterfaceを作ればよいと思う。
VirtualJoystickのDefault設定
data:image/s3,"s3://crabby-images/23121/231212c40e3fe4dc055a58ca3075482052dc31f4" alt=""
Center
Joystickの位置を決める。符号により初期位置が変わる。
左側の値:+の場合は画面左からの距離、-の場合は画面右からの距離。
右側の値:+の場合は画面上からの距離、-の場合は画面下からの距離。
Activate Touch Interfaceを使用する。
data:image/s3,"s3://crabby-images/3f152/3f152f4a3b664300e2b3c25c514724d6f0b7b614" alt=""
FCoreDelegates
// Fill out your copyright notice in the Description page of Project Settings.
#pragma once
#include "CoreMinimal.h"
#include "GameFramework/Pawn.h"
#include "MyMobilePawn.generated.h"
UCLASS()
class METRONOME_API AMyMobilePawn : public APawn
{
GENERATED_BODY()
public:
// Sets default values for this pawn's properties
AMyMobilePawn();
protected:
// Called when the game starts or when spawned
virtual void BeginPlay() override;
public:
// Called every frame
virtual void Tick(float DeltaTime) override;
// Called to bind functionality to input
virtual void SetupPlayerInputComponent(class UInputComponent* PlayerInputComponent) override;
UFUNCTION(BlueprintImplementableEvent)
void OnApplicationWillDeactivate();
UFUNCTION(BlueprintImplementableEvent)
void OnApplicationWillEnterBackground();
UFUNCTION(BlueprintImplementableEvent)
void OnApplicationHasEnteredForeground();
UFUNCTION(BlueprintImplementableEvent)
void OnApplicationHasReactivated();
UFUNCTION(BlueprintImplementableEvent)
void OnApplicationWillTerminate();
};
// Fill out your copyright notice in the Description page of Project Settings.
#include "MyMobilePawn.h"
// Sets default values
AMyMobilePawn::AMyMobilePawn()
{
// Set this pawn to call Tick() every frame. You can turn this off to improve performance if you don't need it.
PrimaryActorTick.bCanEverTick = true;
}
// Called when the game starts or when spawned
void AMyMobilePawn::BeginPlay()
{
FCoreDelegates::ApplicationWillDeactivateDelegate.AddUObject(this, &AMyMobilePawn::OnApplicationWillDeactivate);
FCoreDelegates::ApplicationWillEnterBackgroundDelegate.AddUObject(this, &AMyMobilePawn::OnApplicationWillEnterBackground);
FCoreDelegates::ApplicationHasEnteredForegroundDelegate.AddUObject(this, &AMyMobilePawn::OnApplicationHasEnteredForeground);
FCoreDelegates::ApplicationHasReactivatedDelegate.AddUObject(this, &AMyMobilePawn::OnApplicationHasReactivated);
FCoreDelegates::ApplicationWillTerminateDelegate.AddUObject(this, &AMyMobilePawn::OnApplicationWillTerminate);
Super::BeginPlay();
}
// Called every frame
void AMyMobilePawn::Tick(float DeltaTime)
{
Super::Tick(DeltaTime);
}
// Called to bind functionality to input
void AMyMobilePawn::SetupPlayerInputComponent(UInputComponent* PlayerInputComponent)
{
Super::SetupPlayerInputComponent(PlayerInputComponent);
}
data:image/s3,"s3://crabby-images/6aa8f/6aa8f088d9b2cfbf1d055e83796e6d32708fd1d8" alt=""
雑多な学び
ThirdPersonのC++プロジェクトを参考に、試しに実装しようとした。
InputActionValue.hのincludeがないとエラーが出て諦めた。
.h file
#pragma once
#include "CoreMinimal.h"
#include "GameFramework/Character.h"
#include "InputActionValue.h"
#include "ThirdPersonPracticeCharacter.generated.h"
UCLASS(config=Game)
class AThirdPersonPracticeCharacter : public ACharacter
{
GENERATED_BODY()
/** Camera boom positioning the camera behind the character */
UPROPERTY(VisibleAnywhere, BlueprintReadOnly, Category = Camera, meta = (AllowPrivateAccess = "true"))
class USpringArmComponent* CameraBoom;
/** Follow camera */
UPROPERTY(VisibleAnywhere, BlueprintReadOnly, Category = Camera, meta = (AllowPrivateAccess = "true"))
class UCameraComponent* FollowCamera;
/** MappingContext */
UPROPERTY(EditAnywhere, BlueprintReadOnly, Category = Input, meta = (AllowPrivateAccess = "true"))
class UInputMappingContext* DefaultMappingContext;
/** Jump Input Action */
UPROPERTY(EditAnywhere, BlueprintReadOnly, Category = Input, meta = (AllowPrivateAccess = "true"))
class UInputAction* JumpAction;
/** Move Input Action */
UPROPERTY(EditAnywhere, BlueprintReadOnly, Category = Input, meta = (AllowPrivateAccess = "true"))
class UInputAction* MoveAction;
/** Look Input Action */
UPROPERTY(EditAnywhere, BlueprintReadOnly, Category = Input, meta = (AllowPrivateAccess = "true"))
class UInputAction* LookAction;
public:
AThirdPersonPracticeCharacter();
protected:
/** Called for movement input */
void Move(const FInputActionValue& Value);
/** Called for looking input */
void Look(const FInputActionValue& Value);
protected:
// APawn interface
virtual void SetupPlayerInputComponent(class UInputComponent* PlayerInputComponent) override;
// To add mapping context
virtual void BeginPlay();
public:
/** Returns CameraBoom subobject **/
FORCEINLINE class USpringArmComponent* GetCameraBoom() const { return CameraBoom; }
/** Returns FollowCamera subobject **/
FORCEINLINE class UCameraComponent* GetFollowCamera() const { return FollowCamera; }
};
#include "ThirdPersonPracticeCharacter.h"
#include "Camera/CameraComponent.h"
#include "Components/CapsuleComponent.h"
#include "Components/InputComponent.h"
#include "GameFramework/CharacterMovementComponent.h"
#include "GameFramework/Controller.h"
#include "GameFramework/SpringArmComponent.h"
#include "EnhancedInputComponent.h"
#include "EnhancedInputSubsystems.h"
//////////////////////////////////////////////////////////////////////////
// AThirdPersonPracticeCharacter
AThirdPersonPracticeCharacter::AThirdPersonPracticeCharacter()
{
// Set size for collision capsule
GetCapsuleComponent()->InitCapsuleSize(42.f, 96.0f);
// Don't rotate when the controller rotates. Let that just affect the camera.
bUseControllerRotationPitch = false;
bUseControllerRotationYaw = false;
bUseControllerRotationRoll = false;
// Configure character movement
GetCharacterMovement()->bOrientRotationToMovement = true; // Character moves in the direction of input...
GetCharacterMovement()->RotationRate = FRotator(0.0f, 500.0f, 0.0f); // ...at this rotation rate
// Note: For faster iteration times these variables, and many more, can be tweaked in the Character Blueprint
// instead of recompiling to adjust them
GetCharacterMovement()->JumpZVelocity = 700.f;
GetCharacterMovement()->AirControl = 0.35f;
GetCharacterMovement()->MaxWalkSpeed = 500.f;
GetCharacterMovement()->MinAnalogWalkSpeed = 20.f;
GetCharacterMovement()->BrakingDecelerationWalking = 2000.f;
// Create a camera boom (pulls in towards the player if there is a collision)
CameraBoom = CreateDefaultSubobject<USpringArmComponent>(TEXT("CameraBoom"));
CameraBoom->SetupAttachment(RootComponent);
CameraBoom->TargetArmLength = 400.0f; // The camera follows at this distance behind the character
CameraBoom->bUsePawnControlRotation = true; // Rotate the arm based on the controller
// Create a follow camera
FollowCamera = CreateDefaultSubobject<UCameraComponent>(TEXT("FollowCamera"));
FollowCamera->SetupAttachment(CameraBoom, USpringArmComponent::SocketName); // Attach the camera to the end of the boom and let the boom adjust to match the controller orientation
FollowCamera->bUsePawnControlRotation = false; // Camera does not rotate relative to arm
// Note: The skeletal mesh and anim blueprint references on the Mesh component (inherited from Character)
// are set in the derived blueprint asset named ThirdPersonCharacter (to avoid direct content references in C++)
}
void AThirdPersonPracticeCharacter::BeginPlay()
{
// Call the base class
Super::BeginPlay();
//Add Input Mapping Context
if (APlayerController* PlayerController = Cast<APlayerController>(Controller))
{
if (UEnhancedInputLocalPlayerSubsystem* Subsystem = ULocalPlayer::GetSubsystem<UEnhancedInputLocalPlayerSubsystem>(PlayerController->GetLocalPlayer()))
{
Subsystem->AddMappingContext(DefaultMappingContext, 0);
}
}
}
//////////////////////////////////////////////////////////////////////////
// Input
void AThirdPersonPracticeCharacter::SetupPlayerInputComponent(class UInputComponent* PlayerInputComponent)
{
// Set up action bindings
if (UEnhancedInputComponent* EnhancedInputComponent = CastChecked<UEnhancedInputComponent>(PlayerInputComponent)) {
//Jumping
EnhancedInputComponent->BindAction(JumpAction, ETriggerEvent::Triggered, this, &ACharacter::Jump);
EnhancedInputComponent->BindAction(JumpAction, ETriggerEvent::Completed, this, &ACharacter::StopJumping);
//Moving
EnhancedInputComponent->BindAction(MoveAction, ETriggerEvent::Triggered, this, &AThirdPersonPracticeCharacter::Move);
//Looking
EnhancedInputComponent->BindAction(LookAction, ETriggerEvent::Triggered, this, &AThirdPersonPracticeCharacter::Look);
}
}
void AThirdPersonPracticeCharacter::Move(const FInputActionValue& Value)
{
// input is a Vector2D
FVector2D MovementVector = Value.Get<FVector2D>();
if (Controller != nullptr)
{
// find out which way is forward
const FRotator Rotation = Controller->GetControlRotation();
const FRotator YawRotation(0, Rotation.Yaw, 0);
// get forward vector
const FVector ForwardDirection = FRotationMatrix(YawRotation).GetUnitAxis(EAxis::X);
// get right vector
const FVector RightDirection = FRotationMatrix(YawRotation).GetUnitAxis(EAxis::Y);
// add movement
AddMovementInput(ForwardDirection, MovementVector.Y);
AddMovementInput(RightDirection, MovementVector.X);
}
}
void AThirdPersonPracticeCharacter::Look(const FInputActionValue& Value)
{
// input is a Vector2D
FVector2D LookAxisVector = Value.Get<FVector2D>();
if (Controller != nullptr)
{
// add yaw and pitch input to controller
AddControllerYawInput(LookAxisVector.X);
AddControllerPitchInput(LookAxisVector.Y);
}
}
下の定義が分かりやすい。
EnhancedPlayerInput.cpp(抜粋)
// NOTE: Enum order represents firing priority(lowest to highest) and is important as multiple keys bound to the same action may generate differing trigger event states.
enum class ETriggerEventInternal : uint8
{
None, // No significant trigger state changes occurred
Completed, // Triggering stopped after one or more triggered ticks ETriggerState (Triggered -> None)
Started, // Triggering has begun ETriggerState (None -> Ongoing)
Ongoing, // Triggering is still being processed ETriggerState (Ongoing -> Ongoing)
Canceled, // Triggering has been canceled mid processing ETriggerState (Ongoing -> None)
StartedAndTriggered, // Triggering occurred in a single tick (fires both started and triggered events) ETriggerState (None -> Triggered)
Triggered, // Triggering occurred after one or more processing ticks ETriggerState (Ongoing -> Triggered, Triggered -> Triggered)
};
ETriggerEventInternal UEnhancedPlayerInput::GetTriggerStateChangeEvent(ETriggerState LastTriggerState, ETriggerState NewTriggerState) const
{
// LastTState NewTState Event
// None -> Ongoing = Started
// None -> Triggered = Started + Triggered
// Ongoing -> None = Canceled
// Ongoing -> Ongoing = Ongoing
// Ongoing -> Triggered = Triggered
// Triggered -> Triggered = Triggered
// Triggered -> Ongoing = Ongoing
// Triggered -> None = Completed
switch (LastTriggerState)
{
case ETriggerState::None:
if (NewTriggerState == ETriggerState::Ongoing)
{
return ETriggerEventInternal::Started;
}
else if (NewTriggerState == ETriggerState::Triggered)
{
return ETriggerEventInternal::StartedAndTriggered;
}
break;
case ETriggerState::Ongoing:
if (NewTriggerState == ETriggerState::None)
{
return ETriggerEventInternal::Canceled;
}
else if (NewTriggerState == ETriggerState::Ongoing)
{
return ETriggerEventInternal::Ongoing;
}
else if (NewTriggerState == ETriggerState::Triggered)
{
return ETriggerEventInternal::Triggered;
}
break;
case ETriggerState::Triggered:
if (NewTriggerState == ETriggerState::Triggered)
{
return ETriggerEventInternal::Triggered; // Don't re-raise Started event for multiple completed ticks.
}
else if (NewTriggerState == ETriggerState::Ongoing)
{
return ETriggerEventInternal::Ongoing;
}
else if (NewTriggerState == ETriggerState::None)
{
return ETriggerEventInternal::Completed;
}
break;
}
return ETriggerEventInternal::None;
}
DeltaSecond
GetWorld()->DeltaTimeSeconds
Pawnからの指示で、Directional LightのRotationを変化させる処理。
結局使わなかったが記録しておく。
Directional LightのReferenceはGetするのが難しいので、Event Dispatcherが役に立った。
data:image/s3,"s3://crabby-images/c6699/c669959c13b60bb459e4914911816b39388ff874" alt=""
👇昔のInput(Enhanced Inputではない。)
今でもPlayerInputComponentやSetupPlayerInputComponentの関数は使われているが、このPlayerInputComponentの中身がEnhanced Input Componentになっているのだと思われる。
PawnのProtectedで定義されているvirtual Function。
どこで実行されているのかは調べられなかった。
とにかく、キーのインプットと関数を結び付けるのに使える。
Pawn.h
/** Allows a Pawn to set up custom input bindings. Called upon possession by a PlayerController, using the InputComponent created by CreatePlayerInputComponent(). */
virtual void SetupPlayerInputComponent(UInputComponent* PlayerInputComponent) { /* No bindings by default.*/ }
使用例ThirdPersonのMoveForward
.h file
public:
virtual void ACPPThirdPersonCharacter::SetupPlayerInputComponent(class UInputComponent* PlayerInputComponent) override;
void MoveForward(float Value)
.cpp file
void ACPPThirdPersonCharacter::SetupPlayerInputComponent(class UInputComponent* PlayerInputComponent)
{
PlayerInputComponent->BindAxis("Move Forward / Backward", this, &ACPPThirdPersonCharacter::MoveForward);
}
void ACPPThirdPersonCharacter::MoveForward(float Value)
{
if ((Controller != nullptr) && (Value != 0.0f))
{
// find out which way is forward
const FRotator Rotation = Controller->GetControlRotation();
const FRotator YawRotation(0, Rotation.Yaw, 0);
// get forward vector
const FVector Direction = FRotationMatrix(YawRotation).GetUnitAxis(EAxis::X);
AddMovementInput(Direction, Value);
}
}
BindAxisと結びつける関数はfloat型の引数を持っていないといけないのかも。
” “の中身はProject Settingsのinputで設定した名前。
PlayerInputComponentの中のBindAxisの関数は引数を3つもつ。
詳しくは下記のコードで。
BindAxis
template<class UserClass>
FInputAxisBinding& BindAxis( const FName AxisName, UserClass* Object, typename FInputAxisHandlerSignature::TMethodPtr< UserClass > Func )
{
FInputAxisBinding AB( AxisName );
AB.AxisDelegate.BindDelegate(Object, Func);
AxisBindings.Emplace(MoveTemp(AB));
return AxisBindings.Last();
}
デリゲート(wiki)
VS2022使えるみたい
構造体では初期値を入れていないと、実機プレイする際にエラーがでる。
(=0を入れていなくてエラーがでた。)
UPROPERTY(BlueprintReadOnly, Category = "ConvertRate")
float ConvertedRate = 0;
Project Settingsにおいて、
Always show Touch Interface : true
Use Mouse for Touch : true
UQ3のプロジェクトをスマホの実機プレイをした結果
ボタンをPressした瞬間、他のボタンはReleaseされてしまう。
走れない。
ジャンプと歩く動作が反発して、たまに飛べない。方向転換ができない。
なぜか次のステージに行けない。
スマホは同時入力はできないから工夫が必要そう。
Chaos Destructionが効いた物体が消えている。
セーブは問題なくできている。
World上にPawnがいて、スタート時にそのPawnがPlayerControllerを所有しても、デフォルトPawnはSpawnされるみたい。PlayerStartもしくはカメラ位置から。
SetTextは、メインのウィジェットじゃないと上手くいかなかった。
メインのウィジェットに、USER CREATEDのウィジェットを足していく形でUIを作ろうとしていた。
USER CREATEDのウィジェットのテキストの中身を書き換えようとして失敗した。
詳しい条件などは調べていない。
UEのEditorとVSを閉じる
Exprolerからファイルを削除
.uprojectファイルを右クリック
Generate Visual Studio project files
data:image/s3,"s3://crabby-images/a9f0f/a9f0fe4f4b990776b1d627d6d848818fcb1f92f9" alt=""
スマホ実機プレイで自作プラグインが使えなかった。
RateComponentだけにしたら、同様のMessageがRateComponentにもでた。
対策方法はあるはずだが、しばらくは自作のプラグインは使わない方向で。重いし。
data:image/s3,"s3://crabby-images/535e8/535e8bc876b91af668ad9cd3a6188b2e5f61c716" alt=""
installedをtrueにしたが、変わらなかった。
data:image/s3,"s3://crabby-images/66283/66283f2dbc5ebc862d90d401de9b32aa4595d482" alt=""
Windowsのパッケージ化をしてみても、同様のMessageが出た。
Android Studioの設定。
(Android Studioで、aabファイルの書き換えができないか考えたときの名残の知識。
aabファイルを作った後に書き換えるのは多分無理。)
これはやらなくても大丈夫。プロジェクトごとに設定するものだから、これを設定しても無駄だとわかった。
Android Studioを使ってみたところ、以下のエラーがでていたので修正した。
Installed Build Tools revision 32.0.0 is corrupted. Remove and install again using the SDK Manager.
修正するのはbuild.gradleのandroidの部分。
修正前
android {
compileSdkVersion 32
buildToolsVersion "32.0.0"
defaultConfig {
applicationId "com.example.myapplication"
minSdkVersion 16
targetSdkVersion 32
versionCode 1
versionName "1.0"
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
}
修正後
android {
compileSdkVersion 30
buildToolsVersion "30.0.2"
defaultConfig {
applicationId "com.example.myapplication"
minSdkVersion 16
targetSdkVersion 30
versionCode 1
versionName "1.0"
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
}
左側に電球アイコンがあったので、クリックしてSync Nowを選んだらエラーが消えた。
data:image/s3,"s3://crabby-images/51e5b/51e5b91f37c1d311a03e0354d66a778a29bd0e68" alt=""
これを修正することで、Image Assetを使えるようになった。(アイコンづくりに使えるもの)
👇参考
data:image/s3,"s3://crabby-images/543b9/543b95101ec22dfd7400db203121832002696aa2" alt=""
Win + V
.uprojectの上で右クリック。
data:image/s3,"s3://crabby-images/f8e95/f8e95455fd926f921883bc2c209f69e652561063" alt=""
変更したいVersionを選ぶ
data:image/s3,"s3://crabby-images/0d631/0d63144d747f740eb6e561c1e658f531b0ce3c4d" alt=""
👇ListViewについて
data:image/s3,"s3://crabby-images/e0226/e0226be7f5bbcf61d085c6d9e7db65b402e7cc6d" alt=""
僕はTileViewを作成してみた。作り方は同じ。
初見ならば上の記事を読むべき。処理負荷などについても調べてくれていて面白いし、分かりやすい。
👇これは自分が忘れたときに思い出す用。上の記事をまねただけのもの。モバイルでのドラッグ&ドロップ実装のために調べてみた。
WB_Entry
data:image/s3,"s3://crabby-images/e65fe/e65feebf7bf8fcca62bf5d0f9d19268b16cb5b3b" alt=""
data:image/s3,"s3://crabby-images/ed9da/ed9dadc0a6a0f2d95faad81bc021aad84872fd31" alt=""
data:image/s3,"s3://crabby-images/8ad10/8ad102576289cf4f8d6b714985e12d9ef7625636" alt=""
WB_TileView
data:image/s3,"s3://crabby-images/2b6cd/2b6cdfc9127066ca4ebb300a5b566e878f20385f" alt=""
Construct BP Entry ItemはClassにObjectを入れたもの。Actorを入れると上手くいかなかった記憶アリ。
data:image/s3,"s3://crabby-images/8c5c0/8c5c016206b3b48c99459ad083de2ad81210b0ce" alt=""
data:image/s3,"s3://crabby-images/b9f3f/b9f3fad8513bab9b760e82c034e20c9ef8c22bf9" alt=""
BP_EntryItemは特に処理を記述していないObject。
ただTextを持たせているのみ。
data:image/s3,"s3://crabby-images/23803/23803b726f3d26ef9f194e8b11e5e283cebfbb89" alt=""
フォルダを移動しろってこと?とりあえず無視しておく。
data:image/s3,"s3://crabby-images/f8157/f815755158b094208b71c50728ebb3cbf38af7c1" alt=""
いつかHoudini試してみたい。
data:image/s3,"s3://crabby-images/4642e/4642ee264ea72aae498abec9638078dbac1cb451" alt=""
data:image/s3,"s3://crabby-images/724c2/724c293308d3cda4e690826eb02363beeb1ccedb" alt=""
Twitterしてます
ブログの更新をお知らせ
コメント