#include "dpCollisionChecks.h"
#include "dpEntity.h"
#include "dpShapeBase.h"
#include "dpShapeSphere.h"
#include "dpShapeBox.h"

#include <iostream>
#include <algorithm>

using namespace dpCollisionChecks;

bool dpCollisionChecks::AreColliding(dpEntity* a, dpEntity* b) {
	auto shapeA = a->GetShape();
	auto shapeB = b->GetShape();

	//Sphere to sphere collision
	if (shapeA->GetShapeType() == dpShapeType::Sphere && shapeB->GetShapeType() == dpShapeType::Sphere) {
		return CheckSpheres(a, b);
	}

	return false;
}

bool dpCollisionChecks::CheckSpheres(dpEntity* a, dpEntity* b) {
	if (!a || !b) return false;

	auto posA = a->GetPosition();
	auto distance = Vector3::DistanceSquared(posA, b->GetPosition());

	auto sphereA = static_cast<dpShapeSphere*>(a->GetShape());
	auto sphereB = static_cast<dpShapeSphere*>(b->GetShape());
	const auto radius = sphereA->GetRadius() + sphereB->GetRadius();

	if (distance <= radius * radius)
		return true;

	return false;
}

bool dpCollisionChecks::CheckBoxes(dpEntity* a, dpEntity* b) {
	if (!a || !b) return false;

	auto boxA = static_cast<dpShapeBox*>(a->GetShape());
	auto boxB = static_cast<dpShapeBox*>(b->GetShape());

	const auto& posA = a->GetPosition();
	const auto& posB = b->GetPosition();

	for (const auto& vert : boxA->GetVertices()) {
		if (boxB->IsVertInBox(vert))
			return true;
	}

	/*//Check if we're overlapping on X/Z:
	if ((boxA->GetMaxWidth() >= boxB->GetMinWidth()) && //If our max width is greater than starting X of b
		(boxA->GetMinWidth() <= boxB->GetMaxWidth()) && //If our start x is less than b's max width
		(boxA->GetMaxDepth() >= boxB->GetMinDepth()) &&
		(boxA->GetMinDepth() <= boxB->GetMaxDepth())) {

		//Check if we're in the right height
		if (boxA->GetTop() <= boxB->GetTop() && boxA->GetTop() >= boxB->GetBottom() || //If our top Y is within their minY/maxY bounds
			boxA->GetBottom() <= boxB->GetTop() && boxA->GetBottom() >= boxB->GetBottom()) //Or our bottom Y
			return true; //We definitely are colliding.
	}*/

	/*//Check if we're overlapping on X/Z:
	if ((boxA->GetMaxWidth() >= posB.x) && //If our max width is greater than starting X of b
		(posA.x <= boxB->GetMaxWidth()) && //If our start x is less than b's max width
		(boxA->GetMaxDepth() >= posB.z) &&
		(posA.z <= boxB->GetMaxDepth())) {

		//Check if we're in the right height
		if (boxA->GetTop() <= boxB->GetTop() && boxA->GetTop() >= posB.y || //If our top Y is within their minY/maxY bounds
			posA.y <= boxB->GetTop() && posA.y >= posB.y) //Or our bottom Y
			return true; //We definitely are colliding.
	}*/

	return false;
}

bool dpCollisionChecks::CheckSphereBox(dpEntity* a, dpEntity* b) {
	if (!a || !b) return false;

	NiPoint3 boxPos;
	dpShapeBox* box;

	NiPoint3 spherePos;
	dpShapeSphere* sphere;

	//Figure out which is the box and which is the sphere
	if (a->GetShape()->GetShapeType() == dpShapeType::Box) {
		box = static_cast<dpShapeBox*>(a->GetShape());
		sphere = static_cast<dpShapeSphere*>(b->GetShape());
		boxPos = a->GetPosition();
		spherePos = b->GetPosition();
	} else {
		box = static_cast<dpShapeBox*>(b->GetShape());
		sphere = static_cast<dpShapeSphere*>(a->GetShape());
		boxPos = b->GetPosition();
		spherePos = a->GetPosition();
	}

	//Get closest point from the box to the sphere center by clamping
	float x = std::max(box->m_MinX, std::min(spherePos.x, box->m_MaxX));
	float y = std::max(box->m_MinY, std::min(spherePos.y, box->m_MaxY));
	float z = std::max(box->m_MinZ, std::min(spherePos.z, box->m_MaxZ));

	//Check the distance between that point & our sphere
	float dX = x - spherePos.x;
	float dY = y - spherePos.y;
	float dZ = z - spherePos.z;
	float distanceSquared = (dX * dX) + (dY * dY) + (dZ * dZ);
	const float radius = sphere->GetRadius();

	return distanceSquared < radius* radius;
}