2017-01-20 17:41:49 +11:00

217 lines
8.2 KiB
C#

// Copyright 2014-2017 ClassicalSharp | Licensed under BSD-3
using System;
using ClassicalSharp.Entities;
using ClassicalSharp.GraphicsAPI;
using ClassicalSharp.Map;
using ClassicalSharp.Physics;
using OpenTK;
namespace ClassicalSharp.Model {
/// <summary> Contains a set of quads and/or boxes that describe a 3D object as well as
/// the bounding boxes that contain the entire set of quads and/or boxes. </summary>
public abstract class IModel : IDisposable {
protected Game game;
protected const int quadVertices = 4;
protected const int boxVertices = 6 * quadVertices;
protected RotateOrder Rotate = RotateOrder.ZYX;
internal CachedModel data;
internal bool initalised;
public IModel(Game game) { this.game = game; }
public abstract void CreateParts();
/// <summary> Whether the entity should be slightly bobbed up and down when rendering. </summary>
/// <remarks> e.g. for players when their legs are at the peak of their swing,
/// the whole model will be moved slightly down. </remarks>
public bool Bobbing = true;
/// <summary> Vertical offset from the model's feet/base that the name texture should be drawn at. </summary>
public abstract float NameYOffset { get; }
/// <summary> Vertical offset from the model's feet/base that the model's eye is located. </summary>
public abstract float GetEyeY(Entity entity);
/// <summary> The maximum scale the entity can have (for collisions and rendering). </summary>
public virtual float MaxScale { get { return 2; } }
/// <summary> Scaling factor applied, multiplied by the entity's current model scale. </summary>
public virtual float ShadowScale { get { return 1; } }
/// <summary> Scaling factor applied, multiplied by the entity's current model scale. </summary>
public virtual float NameScale { get { return 1; } }
/// <summary> The size of the bounding box that is used when
/// performing collision detection for this model. </summary>
public abstract Vector3 CollisionSize { get; }
/// <summary> Bounding box that contains this model,
/// assuming that the model is not rotated at all.</summary>
public abstract AABB PickingBounds { get; }
protected Vector3 pos;
protected float cosHead, sinHead;
protected float uScale, vScale, scale;
/// <summary> Returns whether the model should be rendered based on the given entity's position. </summary>
public virtual bool ShouldRender(Entity p, FrustumCulling culling) {
Vector3 pos = p.Position;
AABB bb = PickingBounds;
float maxLen = Math.Max(bb.Width, Math.Max(bb.Height, bb.Length)) * p.ModelScale;
pos.Y += bb.Height / 2; // centre Y coordinate
return culling.SphereInFrustum(pos.X, pos.Y, pos.Z, maxLen);
}
/// <summary> Returns the closest distance of the given entity to the camera. </summary>
public virtual float RenderDistance(Entity p) {
Vector3 pos = p.Position;
AABB bb = PickingBounds;
pos.Y += bb.Height / 2; // centre Y coordinate
Vector3 cPos = game.CurrentCameraPos;
float dx = MinDist(cPos.X - pos.X, bb.Width / 2);
float dy = MinDist(cPos.Y - pos.Y, bb.Height / 2);
float dz = MinDist(cPos.Z - pos.Z, bb.Length / 2);
return dx * dx + dy * dy + dz * dz;
}
static float MinDist(float dist, float extent) {
// Compare min coord, centre coord, and max coord
float dMin = Math.Abs(dist - extent), dMax = Math.Abs(dist + extent);
return Math.Min(Math.Abs(dist), Math.Min(dMin, dMax));
}
/// <summary> Renders the model based on the given entity's position and orientation. </summary>
public void Render(Entity p) {
index = 0;
pos = p.Position;
if (Bobbing) pos.Y += p.anim.bobbingModel;
Vector3I P = Vector3I.Floor(p.EyePosition);
col = game.World.IsValidPos(P) ? game.Lighting.LightCol(P.X, P.Y, P.Z) : game.Lighting.Outside;
uScale = 1 / 64f; vScale = 1 / 32f;
scale = p.ModelScale;
cols[0] = col;
cols[1] = FastColour.ScalePacked(col, FastColour.ShadeYBottom);
cols[2] = FastColour.ScalePacked(col, FastColour.ShadeZ); cols[3] = cols[2];
cols[4] = FastColour.ScalePacked(col, FastColour.ShadeX); cols[5] = cols[4];
float yawDelta = p.HeadYawDegrees - p.YawDegrees;
cosHead = (float)Math.Cos(yawDelta * Utils.Deg2Rad);
sinHead = (float)Math.Sin(yawDelta * Utils.Deg2Rad);
game.Graphics.SetBatchFormat(VertexFormat.P3fT2fC4b);
game.Graphics.PushMatrix();
Matrix4 m = Matrix4.RotateY(-p.YawRadians) * Matrix4.Scale(scale) * Matrix4.Translate(pos.X, pos.Y, pos.Z);
game.Graphics.MultiplyMatrix(ref m);
DrawModel(p);
game.Graphics.PopMatrix();
}
protected abstract void DrawModel(Entity p);
protected void UpdateVB() {
ModelCache cache = game.ModelCache;
game.Graphics.UpdateDynamicIndexedVb(
DrawMode.Triangles, cache.vb, cache.vertices, index);
index = 0;
}
public virtual void Dispose() { }
protected int col;
protected int[] cols = new int[6];
protected internal ModelVertex[] vertices;
protected internal int index, texIndex;
protected int GetTexture(int pTex) {
return pTex > 0 ? pTex : game.ModelCache.Textures[texIndex].TexID;
}
protected BoxDesc MakeBoxBounds(int x1, int y1, int z1, int x2, int y2, int z2) {
return ModelBuilder.MakeBoxBounds(x1, y1, z1, x2, y2, z2);
}
protected BoxDesc MakeRotatedBoxBounds(int x1, int y1, int z1, int x2, int y2, int z2) {
return ModelBuilder.MakeRotatedBoxBounds(x1, y1, z1, x2, y2, z2);
}
protected ModelPart BuildBox(BoxDesc desc) {
return ModelBuilder.BuildBox(this, desc);
}
protected ModelPart BuildRotatedBox(BoxDesc desc) {
return ModelBuilder.BuildRotatedBox(this, desc);
}
protected void DrawPart(ModelPart part) {
VertexP3fT2fC4b vertex = default(VertexP3fT2fC4b);
VertexP3fT2fC4b[] finVertices = game.ModelCache.vertices;
for (int i = 0; i < part.Count; i++) {
ModelVertex v = vertices[part.Offset + i];
vertex.X = v.X; vertex.Y = v.Y; vertex.Z = v.Z;
vertex.Colour = cols[i >> 2];
vertex.U = v.U * uScale; vertex.V = v.V * vScale;
int quadI = i & 3;
if (quadI == 0 || quadI == 3) vertex.V -= 0.01f * vScale;
if (quadI == 2 || quadI == 3) vertex.U -= 0.01f * uScale;
finVertices[index++] = vertex;
}
}
protected void DrawRotate(float angleX, float angleY, float angleZ, ModelPart part) {
DrawRotated(angleX, angleY, angleZ, part, false);
}
protected void DrawHeadRotate(float angleX, float angleY, float angleZ, ModelPart part) {
DrawRotated(angleX, angleY, angleZ, part, true);
}
protected void DrawRotated(float angleX, float angleY, float angleZ, ModelPart part, bool head) {
float cosX = (float)Math.Cos(-angleX), sinX = (float)Math.Sin(-angleX);
float cosY = (float)Math.Cos(-angleY), sinY = (float)Math.Sin(-angleY);
float cosZ = (float)Math.Cos(-angleZ), sinZ = (float)Math.Sin(-angleZ);
float x = part.RotX, y = part.RotY, z = part.RotZ;
VertexP3fT2fC4b vertex = default(VertexP3fT2fC4b);
VertexP3fT2fC4b[] finVertices = game.ModelCache.vertices;
for (int i = 0; i < part.Count; i++) {
ModelVertex v = vertices[part.Offset + i];
v.X -= x; v.Y -= y; v.Z -= z;
float t = 0;
// Rotate locally
if (Rotate == RotateOrder.ZYX) {
t = cosZ * v.X + sinZ * v.Y; v.Y = -sinZ * v.X + cosZ * v.Y; v.X = t; // Inlined RotZ
t = cosY * v.X - sinY * v.Z; v.Z = sinY * v.X + cosY * v.Z; v.X = t; // Inlined RotY
t = cosX * v.Y + sinX * v.Z; v.Z = -sinX * v.Y + cosX * v.Z; v.Y = t; // Inlined RotX
} else if (Rotate == RotateOrder.XZY) {
t = cosX * v.Y + sinX * v.Z; v.Z = -sinX * v.Y + cosX * v.Z; v.Y = t; // Inlined RotX
t = cosZ * v.X + sinZ * v.Y; v.Y = -sinZ * v.X + cosZ * v.Y; v.X = t; // Inlined RotZ
t = cosY * v.X - sinY * v.Z; v.Z = sinY * v.X + cosY * v.Z; v.X = t; // Inlined RotY
}
// Rotate globally
if (head) {
t = cosHead * v.X - sinHead * v.Z; v.Z = sinHead * v.X + cosHead * v.Z; v.X = t; // Inlined RotY
}
vertex.X = v.X + x; vertex.Y = v.Y + y; vertex.Z = v.Z + z;
vertex.Colour = cols[i >> 2];
vertex.U = v.U * uScale; vertex.V = v.V * vScale;
int quadI = i & 3;
if (quadI == 0 || quadI == 3) vertex.V -= 0.01f * vScale;
if (quadI == 2 || quadI == 3) vertex.U -= 0.01f * vScale;
finVertices[index++] = vertex;
}
}
protected enum RotateOrder { ZYX, XZY }
}
}