using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using System.Linq;
///
/// Class which defines blocks around the level
///
[RequireComponent(typeof(BoxCollider))]
public class Block : MonoBehaviour
{
#region Inspector Fields
[SerializeField]
[Tooltip("Offset from the top of the block from which a character should stand")]
private Vector3 VisualOffset = Vector3.zero;
[SerializeField]
[Tooltip("Can this type of block be walked on")]
public bool is_Walkable = true;
[Tooltip("Is this block underwater?")]
public bool isWater = false;
[Tooltip("Is this block at the bottom of a pit?")]
public bool isPit = false;
public bool isCrystals = false;
public bool isRock = false;
public bool isCollectableSpawnable = false;
[Header("Spawn Settings")]
[Tooltip("Can this block be spawned on")]
public bool isSpawnable = false;
[Tooltip("Direction Player is poting at when spawned")]
public Direction SpawnDirection = Direction.Forward;
#endregion InspectorFields
#region Private Functions
///
/// List of current players on this block
///
public Character CurrentPlayer { get; protected set; }
#endregion Private Functions
#region ReadOnly Properties
///
/// Blocks position in global space
///
public Vector3 position { get { return transform.position; } }
///
/// Position character should stand in global space
///
public Vector3 VisualPosition { get { return position + VisualOffset + Vector3.up * 0.5f; } }
#endregion ReadOnly Properties
#region Public Functions
///
/// Is a block valid to walk on
///
/// Layers to check for when checking for blocks above
///
public bool isWalkable(LayerMask layerMask)
{
//checks if there is no block above this one and that this is tagged as walkable
return (is_Walkable && !isBlockAtPosition(position + Vector3.up, 1, layerMask));
}
///
/// Is called when a player moves onto this block
///
/// Should be implemented by a derived class
///
/// Player which moved on to block
///The direction the player moved to get to this block
public virtual IEnumerator OnWalkedOnByPlayer(Character player, Vector3 moveDirection)
{
yield return StartCoroutine(DoPush(player, moveDirection));
}
///
/// Is called when a player moves off of block
///
/// Should be implemented by a derived class
///
/// Player which moved on to block
public virtual void OnLeftByPlayer(Character player)
{
CurrentPlayer = null;
}
///
/// Called to deal with players colliding on this block
///
/// Player which is moving into this block
/// The direction the player moved to get to this block
public virtual IEnumerator DoPush(Character newPlayer, Vector3 moveDirection)
{
if (CurrentPlayer == null)
{
CurrentPlayer = newPlayer;
yield break;
}
Block pushBlock = GetPushLocation(moveDirection, ~CurrentPlayer.Ignore);
if (pushBlock != this)
{
Character oldPlayer = CurrentPlayer;
CurrentPlayer = newPlayer;
yield return StartCoroutine(oldPlayer.MoveToBlock(pushBlock, Character.Animation.Hit, 1));
}
else
{
Block returnBlock = GetPushLocation(-moveDirection, ~newPlayer.Ignore);
if (returnBlock != this)
yield return StartCoroutine(newPlayer.MoveToBlock(returnBlock, Character.Animation.Hit, 1));
}
}
#endregion Public Functions
#region Protected Functions
protected Block GetPushLocation(Vector3 pushDirection, LayerMask ignoreMask)
{
//setting up variables
Vector3 newPosition = position + pushDirection; // position wanted
Block hit; //output of block detection
//if move is obstucted no where to move
if (Block.isBlockAtPosition(newPosition + Vector3.up, 1, ignoreMask))
return this;
//If block at Position is walkable set it to moveTo
if (Block.isBlockAtPosition(newPosition, 1, ignoreMask, out hit) && hit.isWalkable(ignoreMask))
{
return hit;
}
//else if block down one is walkable
else if (Block.isBlockAtPosition(newPosition + Vector3.down, 1, ignoreMask, out hit) && hit.isWalkable(ignoreMask))
{
return hit;
}
return this;
}
#endregion Protected Functions
#region Editor Functions
private void OnDrawGizmos()
{
if (!isSpawnable)
return;
Vector3 DrawPosition = VisualPosition + Vector3.up * 0.4f;
Vector3 Perp = Quaternion.Euler(0, 90, 0) * SpawnDirection.ToVector();
DebugExtensions.DrawCube(DrawPosition, 0.4f, Color.magenta, 0);
//Eyes
Vector3 eyePosition = DrawPosition + SpawnDirection.ToVector() * 0.4f;
DebugExtensions.DrawCube(eyePosition + Perp * 0.2f, 0.1f, Color.magenta, 0);
DebugExtensions.DrawCube(eyePosition - Perp * 0.2f, 0.1f, Color.magenta, 0);
//ears
Vector3 earPosition = DrawPosition + SpawnDirection.ToVector() * 0.2f + Vector3.up * 0.4f;
Vector3 earScale = Quaternion.LookRotation(SpawnDirection.ToVector()) * new Vector3(0.1f, 0.1f, 0.05f);
DebugExtensions.DrawCube(earPosition + Perp * 0.3f, earScale, Color.magenta, 0);
DebugExtensions.DrawCube(earPosition - Perp * 0.3f, earScale, Color.magenta, 0);
}
#endregion Editor Functions
#region Static Functions
///
/// Checks if there is a block at supplied position
///
/// position to check at
/// Scale of block. (should be 1)
/// Layers to check on
/// Block hit
/// if a block is at position
public static bool isBlockAtPosition(Vector3 position, float Scale, LayerMask layerMask, out Block hit)
{
//Turn scale into halfextent and shrink a bit so it doesn't hit bordering blocks
Vector3 halfExtent = Vector3.one * ((Scale - 0.1f) / 2);
//Get every collider which is at position
Collider[] cols = Physics.OverlapBox(position, halfExtent, Quaternion.identity, layerMask);
//Filter colliders for only GameObjects with an Block component
Block[] blocks = cols.Where(p => p.GetComponent() != null).Select(p => p.GetComponent()).ToArray();
//Draw cube, for visuals
DebugExtensions.DrawCube(position, halfExtent, Color.cyan, 1, false);
//if didn't hit anyblocks return false
if (blocks.Length == 0)
{
hit = null;
return false;
}
else
{
//else get the closest block to disered position, (in case we hit mulitple blocks)
hit = Utility.minBy(blocks, p => Vector3.Distance(p.transform.position, position));
return true;
}
}
///
/// Checks if there is a block at supplied position
///
/// position to check at
/// Scale of block. (should be 1)
/// Layers to check on
/// if a block is at position
public static bool isBlockAtPosition(Vector3 position, float scale, LayerMask layerMask)
{
//Return Overloaded function above
Block hit;
return (isBlockAtPosition(position, scale, layerMask, out hit));
}
///
/// Gets the block at a position or creates an airblock if there isn't one there
///
/// position to get block at
/// Scale of block. (should be 1)
/// Layers to check on
/// block at position
public static Block GetOrCreateBlockAtPosition(Vector3 position, float Scale, LayerMask layerMask)
{
Block retVal;
if (isBlockAtPosition(position, Scale, layerMask, out retVal))
return retVal;
GameObject newBlock = Resources.Load("Cube_Air");
newBlock.transform.position = position;
return newBlock.GetComponent();
}
#endregion
}