今まで、公式が出しているスターターアセットでFPSプレイヤーを作っていましたが、「なんだか自分の可能性をつぶしているのではないか…」
なんて思ったりしていたので自作でFPS用のプレイヤーを作る方法をいろいろ調べていき、何とか作れるようになったのでその方法を記述したいと思います。
今回の記事では、「FPSカメラ」と「プレイヤーの基本的な動き」を紹介します。
準備
プレイヤーとなるオブジェクトとその他の細かなものを用意していきます。
まずは、プレイヤーのいろんな要素を入れるための空のオブジェクトを作ります。
data:image/s3,"s3://crabby-images/bcf35/bcf358cbf69d9e64e0d23b7aead2027af86e6a08" alt=""
そうしたら、そのオブジェクトに「Rigidbody」コンポーネントを追加し、プレイヤーのあたり判定の向上のために「Interpolate」を「Interpolate」に、「Collision Detection」を「Continuous」にします。
data:image/s3,"s3://crabby-images/f1bf0/f1bf0a461f3652d059b89da501832c5bdcc8dae2" alt=""
次に、プレイヤーの大事な要素となる空のオブジェクトを三つ用意します。
一つ目は、プレイヤーオブジェクト(自分の場合はCapsule)を先ほどの空のオブジェクトの子オブジェクトとして作ります。
data:image/s3,"s3://crabby-images/30e7d/30e7d5aa3140c0727b53d06637826b8752a26685" alt=""
二つ目は、プレイヤーの向いている方向を知るための空のオブジェクトを一つ子オブジェクトとして作ります。
data:image/s3,"s3://crabby-images/27356/2735602f5f7dfcfb0b660d87217fb6a83283a251" alt=""
最後に、カメラの位置となる空のオブジェクトを作ります。
data:image/s3,"s3://crabby-images/a17ed/a17ed0e757fb47e766d2e3c2422d7259d0511d56" alt=""
位置をプレイヤーの頭あたりに持っていきます。
data:image/s3,"s3://crabby-images/58633/586338983efb0fd4128c366251a7a84b19439ccf" alt=""
次にプレイヤーとは別にカメラ用に空のオブジェクトを用意します。
data:image/s3,"s3://crabby-images/7c8dc/7c8dcb90d0883fcb643426b5fd55c0d0ae073916" alt=""
その中に「Main camera」を入れましょう。
data:image/s3,"s3://crabby-images/1e667/1e6677d4f3678ac91da2dacc8f6451054f541e3d" alt=""
「Main Camera」を子オブジェクトにしたら「Position」をすべて「0」にしておきます。
data:image/s3,"s3://crabby-images/b46b5/b46b5fa111987e0991e3c205820fb1c7ccd6a32d" alt=""
ポイント:カメラをプレイヤーの中に入れずに別の空のオブジェクトに入れているのは、コライダーを持っているオブジェクトの中に入れてしまうとカメラがおかしな挙動をしてしまうかららしい…
次に、地面と接しているのかを判断するために地面となるオブジェクトにレイヤーを設定します。
インスペクター右上のLayerのDefault>Add layerをクリックします。
data:image/s3,"s3://crabby-images/d0104/d0104a39bce7b7401f15bcb83cf6af3792402c58" alt=""
どこか空白になっているところに「Ground」と、新しいレイヤーを作ります。
data:image/s3,"s3://crabby-images/98e5f/98e5fbc882cc537daaea323958318e5921881535" alt=""
そうしたら地面となるオブジェクトに先ほど作ったレイヤーを設定します。
data:image/s3,"s3://crabby-images/0a1a2/0a1a2a754df6145cf3fb1daeb9289a05dba3ffa1" alt=""
これで準備は完了です。
カメラを動かせるようにしよう
それでは、マウスでカメラを動かせるようにしましょう。
まずは、カメラをプレイヤーの頭の位置に固定したいのでこのスクリプトを「Main Camera」の親オブジェクトにアタッチします。
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
public class CameraPos : MonoBehaviour
{
public Transform cameraPos;
void Update()
{
transform.position = cameraPos.position;
}
}
data:image/s3,"s3://crabby-images/68ad0/68ad0e44241b041508cfde5bf2f60c170770e41f" alt=""
「Camera Pos」の中にプレイヤーで作ったカメラの位置となる子オブジェクトを入れます。
data:image/s3,"s3://crabby-images/58ada/58ada96fc54fad856d931ed709a82b6461f47c0a" alt=""
次に、カメラを動かすこのスクリプトを「Main Camera」にアタッチします。
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
public class PlayerCam : MonoBehaviour
{
public float sensX;
public float sensY;
public Transform orientation;
public Transform CamHolder;
float xRotation;
float yRotation;
void Start()
{
//マウスポインターを真ん中に固定し見えなくする
Cursor.lockState = CursorLockMode.Locked;
Cursor.visible = false;
}
// Update is called once per frame
void Update()
{
//マウスの入力を取得
float mouseX = Input.GetAxisRaw("Mouse X") * Time.deltaTime * sensX;
float mouseY = Input.GetAxisRaw("Mouse Y") * Time.deltaTime * sensY;
yRotation += mouseX;
xRotation -= mouseY;
//Y軸だけ視点に上限を設ける
xRotation = Mathf.Clamp(xRotation, -90, 90);
//カメラとプレイヤーの向きを動かす
CamHolder.rotation = Quaternion.Euler(xRotation, yRotation, 0);
orientation.rotation = Quaternion.Euler(0, yRotation, 0);
}
}
data:image/s3,"s3://crabby-images/5a1dd/5a1dde6402c573c361360ca2dadef7a72185443a" alt=""
「Orientation」にはプレイヤーの中にあるプレイヤーの向いている方向がわかるオブジェクトを、「cam Holder」には「Main camera」を入れた親オブジェクトをいれます。
data:image/s3,"s3://crabby-images/f3a68/f3a68208aa1957f015351ea7e18de833f10c2ce7" alt=""
そして「sensX」と「sensY」はカメラの感度のようなものになります。
適当に両方とも「400」にします。
data:image/s3,"s3://crabby-images/cee0f/cee0f46b0b702ef5cc29faef8713685c66f5b2ca" alt=""
これで実行してみましょう。
FPSカメラが出来上がりました!
プレイヤーが動くようにしよう
それでは、プレイヤーが動けるようにしましょう。
このスクリプトをプレイヤーにアタッチしましょう。
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
public class PlayerMovement : MonoBehaviour
{
public float moveSpeed;
public float groundDrag;
public float playerheight;
public LayerMask Ground;
bool grounded;
public Transform orientation;
float HorizontalInput;
float VerticalInput;
Vector3 moveDirection;
Rigidbody rb;
void Start()
{
rb = GetComponent<Rigidbody>();
rb.freezeRotation = true;
}
void Update()
{
//地面と接しているかを判断
grounded = Physics.Raycast(transform.position, Vector3.down, playerheight * 0.5f + 0.2f, Ground);
//接している場合は、設定した減速値を代入しプレイヤーを滑りにくくする
if (grounded)
rb.drag = groundDrag;
else
rb.drag = 0;
ProcessInput();
SpeedControl();
}
private void FixedUpdate()
{
movePlayer();
}
private void ProcessInput()
{
//入力を取得
HorizontalInput = Input.GetAxisRaw("Horizontal");
VerticalInput = Input.GetAxisRaw("Vertical");
}
private void movePlayer()
{
//向いている方向に進む
moveDirection = orientation.forward * VerticalInput + orientation.right * HorizontalInput;
rb.AddForce(moveDirection.normalized * moveSpeed * 10f, ForceMode.Force);
}
private void SpeedControl()
{
//プレイヤーのスピードを制限
Vector3 flatVel = new Vector3(rb.velocity.x, 0, rb.velocity.z);
if (flatVel.magnitude > moveSpeed)
{
Vector3 limitedVel = flatVel.normalized * moveSpeed;
rb.velocity = new Vector3(limitedVel.x, rb.velocity.y, limitedVel.z);
}
}
}
このようになっていますね。
data:image/s3,"s3://crabby-images/4b2f3/4b2f3c69218056285235d9166fa9a9104f6265b4" alt=""
値をとりあえずこのように設定しましょう。
Move Speed:プレイヤーの動くスピード
Ground Drag:地面に接しているときの減速する度合い ←これないとプレイヤが滑っているような挙動になる
Playerheight:プレイヤーの高さ
Ground:最初の方で作った地面を識別するためのレイヤーを入れよう
Orientation:プレイヤーの中にある向いている方向を知るための空のオブジェクトを入れる
data:image/s3,"s3://crabby-images/b0ebe/b0ebe5a497ee864667bce3e876307438c2c8bcae" alt=""
data:image/s3,"s3://crabby-images/48f79/48f7905a24a7a25c5e9760faa7a1e92d56a811ff" alt=""
これで実行してみますと
プレイヤーが動くようになりました!
まとめ
スターターアセットを使わずに自分でFPS用のプレイヤースクリプトを使うことでいろいろいじくれそうな気がしたので、ジャンプやしゃがみといったモーションも追加していきたいと思います。
コメント