data:image/s3,"s3://crabby-images/a9e82/a9e823f7c2dce4bf4a92b5c55fdcf86daf6071ae" alt=""
Unityで簡単に雷を作ってみました。Planeに画像を表示してアニメーションさせました。
data:image/s3,"s3://crabby-images/36da2/36da2191edf12e30b96620ee46f2708a993c106a" alt=""
まず適当に雷の画像を作りました。
data:image/s3,"s3://crabby-images/8e0aa/8e0aa134d5247bf960f59fd30a6c05c147e5eb4e" alt=""
これをGIMP2で開いて、白いレイヤーを追加して、そのレイヤーマスクにこの画像をコピペします。元の画像を非表示にすると、黒い部分な透明になります。この画像と元の画像をUnityにインポートしました。
data:image/s3,"s3://crabby-images/e82db/e82dbfb42fc8bb047e71e0e9b30e9727559d097b" alt=""
data:image/s3,"s3://crabby-images/14b7c/14b7c11d6b81e553eb5811167d00052a900eac57" alt=""
Unityで雷用のマテリアルを作って、Transparentタイプにし、Preserve specular lightingのチェックを外します。Base Mapに透明にした画像を、Emissive Colorに元の画像をアタッチしました。
data:image/s3,"s3://crabby-images/eab75/eab75a300dccd123455e09da2cfedc6fafc5daa7" alt=""
Emissive Colorの横のカラーピッカーでIntensityを変えると、雷の光具合を変えられます。
data:image/s3,"s3://crabby-images/173d7/173d74f3a4120f738dbe93ba895d542ac15950a1" alt=""
光らないだけでは白く表示されたままなので、雷を消すときはBase Mapの横のカラーピッカーで透明にします。
data:image/s3,"s3://crabby-images/b17d7/b17d74257075e1c9be5d8fa6aef69a5bf6b763bc" alt=""
このマテリアルを雷用のPlaneに付けます。それを空のゲームオブジェクトの子にして、Planeの子にポイントライトを置きました。空のゲームオブジェクトのポジションを0にしてから子オブジェクトを設定して、Planeやライトの位置も0にしておきます。
data:image/s3,"s3://crabby-images/06c2f/06c2fb53c337238287cc12d93e0bffa725b6d3c6" alt=""
ポイントライトの明るさはLightコンポーネントのIntensityで変えます。Rangeは大きめにしておきました。
data:image/s3,"s3://crabby-images/5bd2e/5bd2e297f4539ad7dc7be9b05c2d1647d748ad78" alt=""
ルートのオブジェクトを選択して、AnimationウィンドウでCreateを押すと、Animatorコンポーネントと最初のアニメーションが作られます。スクリプトも付けて、自分を破壊するメソッドを定義しておきます。
data:image/s3,"s3://crabby-images/fbe4c/fbe4c119d92fdd8de78a13e0d879f997f457fc6b" alt=""
// ...
void DestroyThunder()
{
Destroy(gameObject);
}
}
Animationウィンドウでは、Planeの不透明度やエミッションの強さ、ポイントライトの強さなどのアニメーションを付けます。アニメーションの最後のほうにシグナルを追加します。
data:image/s3,"s3://crabby-images/a16d7/a16d74963cb636f4856fe34b3931eccbbfdb466d" alt=""
追加したシグナルを選択して、インスペクタで破壊メソッドを選択すると、このタイミングでこのメソッドが呼ばれて雷が消えます。
スクリプトではStartメソッドでプレイヤーの位置に応じて雷の向きを変えます。
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
public class Thunder : MonoBehaviour
{
private void Start()
{
// プレイヤーと同じ高さにする
Vector3 pos = transform.position;
pos.y = UnityStandardAssets.Characters.FirstPerson.FirstPersonController.GetInstance().transform.position.y;
transform.position = pos;
// プレイヤーの方を向かせる
transform.LookAt(UnityStandardAssets.Characters.FirstPerson.FirstPersonController.GetInstance().transform);
// 爆発
foreach(GameObject cube in GameObject.FindGameObjectsWithTag("Cube"))
{
Cube cb = cube.GetComponent<Cube>();
if(cb != null)
{
cb.Blow(pos);
}
}
// 少し上に持ち上げる
pos.y += 5f;
transform.position = pos;
}
void DestroyThunder()
{
Destroy(gameObject);
}
}
プレイヤーの方を向かせるためにTransform.LookAtメソッドを使っています。雷とプレイヤーの高さが揃っていないときに呼ぶと、雷が前後に傾きます。
data:image/s3,"s3://crabby-images/908cc/908cc348cd3352fcf65729b041c116e07d262746" alt=""
なので、まず高さを合わせてからプレイヤーの方を向かせた後に、少し高い位置に移動しています。LookAtでは、ゲームオブジェクトのローカルのZ座標が引数のトランスフォームの方へ向くように回転します。
なので、空のゲームオブジェクトのZ座標の方から見た時に、Planeの雷画像が正面に見えるようにPlaneを回転させています。
data:image/s3,"s3://crabby-images/dfb4b/dfb4b290cbf1f9dacad227f476341116aad4b94b" alt=""
data:image/s3,"s3://crabby-images/8ac99/8ac99257b602827a9829744797cbc1b743e96a14" alt=""
高い位置へ移動する前に、その位置を中心とした爆発の力を周りのオブジェクトに加えます。シーンに複数のCubeオブジェクトを配置して、全てにRigidbodyとCubeタグ、スクリプトを付けています。
data:image/s3,"s3://crabby-images/3bf06/3bf06c079ab44d82f7881c9cb37a1ca1fca89d78" alt=""
力を与えるときはタグで検索して一つひとつのCubeクラスの吹き飛ぶメソッドを呼んでいます。その中でRigidbody.AddExplosionForceを呼びます。
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
public class Cube : MonoBehaviour
{
// Start is called before the first frame update
void Start()
{
}
public void Blow(Vector3 centerPosition)
{
Rigidbody rb = GetComponent<Rigidbody>();
if (rb != null)
{
rb.AddExplosionForce(10f, centerPosition, 15f, 20f, ForceMode.Impulse);
}
}
}
雷を出現させるスクリプトを作って、雷のプレハブと出現位置のトランスフォームをアタッチします。プレハブをインスタンス化すると自動でアニメーションが再生されて、終わると破壊されます。
data:image/s3,"s3://crabby-images/1203c/1203c3480c1a25923969f6c504053a8cb4ad6e4c" alt=""
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
public class ThunderFactory : MonoBehaviour
{
[SerializeField] Thunder thunder; // 雷のプレハブ
[SerializeField] Transform thunderPoint; // 雷の出現位置
// Start is called before the first frame update
void Start()
{
}
// Update is called once per frame
void Update()
{
if(Input.GetKeyDown(KeyCode.T))
{
Instantiate(thunder.gameObject, thunderPoint.position, Quaternion.identity);
}
}
}
これで簡単に雷エフェクトが作れました。
data:image/s3,"s3://crabby-images/bc5f5/bc5f57f2fa7aad9911d6b5b976acf64af418df67" alt=""