// Copyright 2014-2017 ClassicalSharp | Licensed under BSD-3 using System; using ClassicalSharp.Model; using ClassicalSharp.Physics; using OpenTK; #if USE16_BIT using BlockID = System.UInt16; #else using BlockID = System.Byte; #endif namespace ClassicalSharp.Entities { /// Contains a model, along with position, velocity, and rotation. /// May also contain other fields and properties. public abstract partial class Entity { AABB modelAABB; /// Returns the bounding box that contains the model, assuming it is not rotated. public AABB PickingBounds { get { return modelAABB.Offset(Position); } } /// Bounding box of the model that collision detection /// is performed with, in world coordinates. public AABB Bounds { get { return AABB.Make(Position, Size); } } /// Determines whether any of the blocks that intersect the /// bounding box of this entity satisfy the given condition. public bool TouchesAny(Predicate condition) { return TouchesAny(Bounds, condition); } /// Determines whether any of the blocks that intersect the /// given bounding box satisfy the given condition. public bool TouchesAny(AABB bounds, Predicate condition) { Vector3I bbMin = Vector3I.Floor(bounds.Min); Vector3I bbMax = Vector3I.Floor(bounds.Max); BlockInfo info = game.BlockInfo; AABB blockBB = default(AABB); // Order loops so that we minimise cache misses for (int y = bbMin.Y; y <= bbMax.Y; y++) for (int z = bbMin.Z; z <= bbMax.Z; z++) for (int x = bbMin.X; x <= bbMax.X; x++) { if (!game.World.IsValidPos(x, y, z)) continue; BlockID block = game.World.GetBlock(x, y, z); blockBB.Min = new Vector3(x, y, z) + info.MinBB[block]; blockBB.Max = new Vector3(x, y, z) + info.MaxBB[block]; if (!blockBB.Intersects(bounds)) continue; if (condition(block)) return true; } return false; } /// Determines whether any of the blocks that intersect the /// bounding box of this entity are rope. public bool TouchesAnyRope() { AABB bounds = Bounds; bounds.Max.Y += 0.5f/16f; return TouchesAny(bounds, b => b == Block.Rope); } /// Constant offset used to avoid floating point roundoff errors. public const float Adjustment = 0.001f; static readonly Vector3 liqExpand = new Vector3(0.25f/16f, 0/16f, 0.25f/16f); // If liquid block above, leave height same // otherwise reduce water BB height by 0.5 blocks bool TouchesAnyLiquid(AABB bounds, BlockID block1, BlockID block2) { Vector3I bbMin = Vector3I.Floor(bounds.Min); Vector3I bbMax = Vector3I.Floor(bounds.Max); int height = game.World.Height; BlockInfo info = game.BlockInfo; AABB blockBB = default(AABB); // Order loops so that we minimise cache misses for (int y = bbMin.Y; y <= bbMax.Y; y++) for (int z = bbMin.Z; z <= bbMax.Z; z++) for (int x = bbMin.X; x <= bbMax.X; x++) { if (!game.World.IsValidPos(x, y, z)) continue; BlockID block = game.World.GetBlock(x, y, z); BlockID below = (y - 1) < 0 ? Block.Air : game.World.GetBlock(x, y - 1, z); BlockID above = (y + 1) >= height ? Block.Air : game.World.GetBlock(x, y + 1, z); // TODO: use recording to find right constants when I have more time blockBB.Min = new Vector3(x, y, z) + info.MinBB[block]; blockBB.Max = new Vector3(x, y, z) + info.MaxBB[block]; //if (game.BlockInfo.Collide[below] != CollideType.SwimThrough) // min.Y += 4/16f; //if (game.BlockInfo.Collide[above] != CollideType.SwimThrough) // max.Y -= 4/16f; if (!blockBB.Intersects(bounds)) continue; if (block == block1 || block == block2) return true; } return false; } /// Determines whether any of the blocks that intersect the /// bounding box of this entity are lava or still lava. public bool TouchesAnyLava() { // NOTE: Original classic client uses offset (so you can only climb up // alternating liquid-solid elevators on two sides) AABB bounds = Bounds.Offset(liqExpand); return TouchesAnyLiquid(bounds, Block.Lava, Block.StillLava); } /// Determines whether any of the blocks that intersect the /// bounding box of this entity are water or still water. public bool TouchesAnyWater() { AABB bounds = Bounds.Offset(liqExpand); return TouchesAnyLiquid(bounds, Block.Water, Block.StillWater); } } }