2015-10-06 20:15:48 +11:00
|
|
|
|
using System;
|
|
|
|
|
using ClassicalSharp.GraphicsAPI;
|
|
|
|
|
using ClassicalSharp.Model;
|
2015-10-08 16:47:53 +11:00
|
|
|
|
using OpenTK;
|
2015-10-06 20:15:48 +11:00
|
|
|
|
|
|
|
|
|
namespace ClassicalSharp {
|
|
|
|
|
|
|
|
|
|
public static class IsometricBlockDrawer {
|
|
|
|
|
|
|
|
|
|
static BlockInfo info;
|
|
|
|
|
static ModelCache cache;
|
2016-01-13 23:12:03 +11:00
|
|
|
|
static IGraphicsApi api;
|
|
|
|
|
static TerrainAtlas1D atlas;
|
2015-10-06 20:15:48 +11:00
|
|
|
|
static int index;
|
|
|
|
|
static float scale;
|
2015-12-23 21:58:05 +11:00
|
|
|
|
static Vector3 minBB, maxBB;
|
|
|
|
|
const float invElemSize = TerrainAtlas2D.invElementSize;
|
2015-12-28 22:12:23 +11:00
|
|
|
|
static bool fullBright;
|
2015-10-06 20:15:48 +11:00
|
|
|
|
|
2015-10-07 16:16:39 +11:00
|
|
|
|
static FastColour colNormal, colXSide, colZSide, colYBottom;
|
2015-10-08 16:47:53 +11:00
|
|
|
|
static float cosX, sinX, cosY, sinY;
|
2015-10-07 16:16:39 +11:00
|
|
|
|
static IsometricBlockDrawer() {
|
|
|
|
|
colNormal = FastColour.White;
|
|
|
|
|
FastColour.GetShaded( colNormal, ref colXSide, ref colZSide, ref colYBottom );
|
2015-10-08 16:47:53 +11:00
|
|
|
|
|
|
|
|
|
cosX = (float)Math.Cos( 26.565f * Utils.Deg2Rad );
|
|
|
|
|
sinX = (float)Math.Sin( 26.565f * Utils.Deg2Rad );
|
|
|
|
|
cosY = (float)Math.Cos( -45f * Utils.Deg2Rad );
|
|
|
|
|
sinY = (float)Math.Sin( -45f * Utils.Deg2Rad );
|
2015-10-07 16:16:39 +11:00
|
|
|
|
}
|
|
|
|
|
|
2015-10-06 20:15:48 +11:00
|
|
|
|
public static void Draw( Game game, byte block, float size, float x, float y ) {
|
|
|
|
|
info = game.BlockInfo;
|
|
|
|
|
cache = game.ModelCache;
|
2016-01-13 23:12:03 +11:00
|
|
|
|
atlas = game.TerrainAtlas1D;
|
2015-12-23 21:58:05 +11:00
|
|
|
|
minBB = info.MinBB[block];
|
|
|
|
|
maxBB = info.MaxBB[block];
|
2015-12-28 22:12:23 +11:00
|
|
|
|
fullBright = info.FullBright[block];
|
2015-12-23 21:58:05 +11:00
|
|
|
|
if( info.IsSprite[block] ) {
|
|
|
|
|
minBB = Vector3.Zero; maxBB = Vector3.One;
|
|
|
|
|
}
|
2015-12-30 21:42:20 +11:00
|
|
|
|
if( info.IsAir[block] ) return;
|
2015-10-06 20:15:48 +11:00
|
|
|
|
index = 0;
|
2016-01-13 23:12:03 +11:00
|
|
|
|
api = game.Graphics;
|
2015-12-30 21:42:20 +11:00
|
|
|
|
|
2015-11-09 19:53:24 +11:00
|
|
|
|
// isometric coords size: cosY * -scale - sinY * scale
|
2015-11-30 08:08:32 +11:00
|
|
|
|
// we need to divide by (2 * cosY), as the calling function expects size to be in pixels.
|
2016-01-13 23:12:03 +11:00
|
|
|
|
scale = size / (2 * cosY);
|
2015-10-08 16:47:53 +11:00
|
|
|
|
// screen to isometric coords (cos(-x) = cos(x), sin(-x) = -sin(x))
|
2016-01-13 23:12:03 +11:00
|
|
|
|
pos.X = x; pos.Y = y; pos.Z = 0;
|
2015-10-08 16:47:53 +11:00
|
|
|
|
pos = Utils.RotateY( Utils.RotateX( pos, cosX, -sinX ), cosY, -sinY );
|
2015-10-06 20:15:48 +11:00
|
|
|
|
|
|
|
|
|
if( info.IsSprite[block] ) {
|
2015-10-29 06:28:23 +11:00
|
|
|
|
XQuad( block, 0f, TileSide.Right );
|
|
|
|
|
ZQuad( block, 0f, TileSide.Back );
|
2015-10-06 20:15:48 +11:00
|
|
|
|
} else {
|
2015-12-23 21:58:05 +11:00
|
|
|
|
XQuad( block, Make( maxBB.X ), TileSide.Left );
|
|
|
|
|
ZQuad( block, Make( minBB.Z ), TileSide.Back );
|
|
|
|
|
YQuad( block, Make( maxBB.Y ), TileSide.Top );
|
2015-10-06 20:15:48 +11:00
|
|
|
|
}
|
2015-10-07 19:39:20 +11:00
|
|
|
|
|
2016-01-13 23:12:03 +11:00
|
|
|
|
if( index == 0 ) return;
|
|
|
|
|
if( atlas.TexIds[texIndex] != lastTexId ) {
|
|
|
|
|
lastTexId = atlas.TexIds[texIndex];
|
|
|
|
|
api.BindTexture( lastTexId );
|
|
|
|
|
}
|
2015-10-07 19:39:20 +11:00
|
|
|
|
for( int i = 0; i < index; i++ )
|
|
|
|
|
TransformVertex( ref cache.vertices[i] );
|
2016-01-13 23:12:03 +11:00
|
|
|
|
api.UpdateDynamicIndexedVb( DrawMode.Triangles, cache.vb, cache.vertices, index, index * 6 / 4 );
|
2015-10-07 19:39:20 +11:00
|
|
|
|
}
|
2015-10-06 20:15:48 +11:00
|
|
|
|
|
2015-10-07 19:39:20 +11:00
|
|
|
|
static Vector3 pos = Vector3.Zero;
|
2015-10-29 06:28:23 +11:00
|
|
|
|
static void YQuad( byte block, float y, int side ) {
|
2016-01-13 23:12:03 +11:00
|
|
|
|
int texLoc = info.GetTextureLoc( block, side );
|
|
|
|
|
TextureRec rec = atlas.GetTexRec( texLoc, 1, out texIndex );
|
|
|
|
|
FlushIfNotSame();
|
|
|
|
|
FastColour col = colNormal;
|
2015-12-23 21:58:05 +11:00
|
|
|
|
|
2016-01-17 16:40:14 +11:00
|
|
|
|
float vOrigin = (texLoc % atlas.elementsPerAtlas1D) * atlas.invElementSize;
|
2016-01-13 23:12:03 +11:00
|
|
|
|
rec.U1 = minBB.X; rec.U2 = maxBB.X;
|
|
|
|
|
rec.V1 = vOrigin + minBB.Z * atlas.invElementSize;
|
|
|
|
|
rec.V2 = vOrigin + maxBB.Z * atlas.invElementSize * 15.99f/16f;
|
2015-10-06 20:15:48 +11:00
|
|
|
|
|
2015-12-23 21:58:05 +11:00
|
|
|
|
cache.vertices[index++] = new VertexPos3fTex2fCol4b( pos.X + Make( minBB.X ), pos.Y + y,
|
2016-01-08 10:35:14 +11:00
|
|
|
|
pos.Z + Make( minBB.Z ), rec.U1, rec.V1, col );
|
2015-12-23 21:58:05 +11:00
|
|
|
|
cache.vertices[index++] = new VertexPos3fTex2fCol4b( pos.X + Make( maxBB.X ), pos.Y + y,
|
|
|
|
|
pos.Z + Make( minBB.Z ), rec.U2, rec.V1, col );
|
2016-01-08 10:35:14 +11:00
|
|
|
|
cache.vertices[index++] = new VertexPos3fTex2fCol4b( pos.X + Make( maxBB.X ), pos.Y + y,
|
|
|
|
|
pos.Z + Make( maxBB.Z ), rec.U2, rec.V2, col );
|
|
|
|
|
cache.vertices[index++] = new VertexPos3fTex2fCol4b( pos.X + Make( minBB.X ), pos.Y + y,
|
|
|
|
|
pos.Z + Make( maxBB.Z ), rec.U1, rec.V2, col );
|
2015-10-06 20:15:48 +11:00
|
|
|
|
}
|
|
|
|
|
|
2015-10-29 06:28:23 +11:00
|
|
|
|
static void ZQuad( byte block, float z, int side ) {
|
2016-01-13 23:12:03 +11:00
|
|
|
|
int texLoc = info.GetTextureLoc( block, side );
|
|
|
|
|
TextureRec rec = atlas.GetTexRec( texLoc, 1, out texIndex );
|
|
|
|
|
FlushIfNotSame();
|
2015-12-28 22:12:23 +11:00
|
|
|
|
FastColour col = fullBright ? colNormal : colZSide;
|
2015-10-06 20:15:48 +11:00
|
|
|
|
|
2016-01-17 16:40:14 +11:00
|
|
|
|
float vOrigin = (texLoc % atlas.elementsPerAtlas1D) * atlas.invElementSize;
|
2016-01-13 23:12:03 +11:00
|
|
|
|
rec.U1 = minBB.X; rec.U2 = maxBB.X;
|
2016-01-17 16:40:14 +11:00
|
|
|
|
rec.V1 = vOrigin + (1 - minBB.Y) * atlas.invElementSize;
|
|
|
|
|
rec.V2 = vOrigin + (1 - maxBB.Y) * atlas.invElementSize * 15.99f/16f;
|
2015-12-23 21:58:05 +11:00
|
|
|
|
|
|
|
|
|
cache.vertices[index++] = new VertexPos3fTex2fCol4b( pos.X + Make( minBB.X ), pos.Y + Make( maxBB.Y ),
|
2016-01-17 16:40:14 +11:00
|
|
|
|
pos.Z + z, rec.U2, rec.V2, col );
|
|
|
|
|
cache.vertices[index++] = new VertexPos3fTex2fCol4b( pos.X + Make( minBB.X ), pos.Y + Make( minBB.Y ),
|
2015-12-23 21:58:05 +11:00
|
|
|
|
pos.Z + z, rec.U2, rec.V1, col );
|
|
|
|
|
cache.vertices[index++] = new VertexPos3fTex2fCol4b( pos.X + Make( maxBB.X ), pos.Y + Make( minBB.Y ),
|
2016-01-17 16:40:14 +11:00
|
|
|
|
pos.Z + z, rec.U1, rec.V1, col );
|
|
|
|
|
cache.vertices[index++] = new VertexPos3fTex2fCol4b( pos.X + Make( maxBB.X ), pos.Y + Make( maxBB.Y ),
|
2015-12-27 21:22:11 +11:00
|
|
|
|
pos.Z + z, rec.U1, rec.V2, col );
|
2015-12-23 21:58:05 +11:00
|
|
|
|
}
|
2015-10-06 20:15:48 +11:00
|
|
|
|
|
2015-10-29 06:28:23 +11:00
|
|
|
|
static void XQuad( byte block, float x, int side ) {
|
2016-01-13 23:12:03 +11:00
|
|
|
|
int texLoc = info.GetTextureLoc( block, side );
|
|
|
|
|
TextureRec rec = atlas.GetTexRec( texLoc, 1, out texIndex );
|
|
|
|
|
FlushIfNotSame();
|
2015-12-28 22:12:23 +11:00
|
|
|
|
FastColour col = fullBright ? colNormal : colXSide;
|
2015-10-06 20:15:48 +11:00
|
|
|
|
|
2016-01-17 16:40:14 +11:00
|
|
|
|
float vOrigin = (texLoc % atlas.elementsPerAtlas1D) * atlas.invElementSize;
|
2016-01-13 23:12:03 +11:00
|
|
|
|
rec.U1 = minBB.Z; rec.U2 = maxBB.Z;
|
2016-01-17 16:40:14 +11:00
|
|
|
|
rec.V1 = vOrigin + (1 - minBB.Y) * atlas.invElementSize;
|
|
|
|
|
rec.V2 = vOrigin + (1 - maxBB.Y) * atlas.invElementSize * 15.99f/16f;
|
2015-12-23 21:58:05 +11:00
|
|
|
|
|
|
|
|
|
cache.vertices[index++] = new VertexPos3fTex2fCol4b( pos.X + x, pos.Y + Make( maxBB.Y ),
|
2016-01-17 16:40:14 +11:00
|
|
|
|
pos.Z + Make( minBB.Z ), rec.U2, rec.V2, col );
|
|
|
|
|
cache.vertices[index++] = new VertexPos3fTex2fCol4b( pos.X + x, pos.Y + Make( minBB.Y ),
|
2015-12-27 21:22:11 +11:00
|
|
|
|
pos.Z + Make( minBB.Z ), rec.U2, rec.V1, col );
|
2015-12-23 21:58:05 +11:00
|
|
|
|
cache.vertices[index++] = new VertexPos3fTex2fCol4b( pos.X + x, pos.Y + Make( minBB.Y ),
|
2016-01-17 16:40:14 +11:00
|
|
|
|
pos.Z + Make( maxBB.Z ), rec.U1, rec.V1, col );
|
|
|
|
|
cache.vertices[index++] = new VertexPos3fTex2fCol4b( pos.X + x, pos.Y + Make( maxBB.Y ),
|
2015-12-27 21:22:11 +11:00
|
|
|
|
pos.Z + Make( maxBB.Z ), rec.U1, rec.V2, col );
|
2015-10-06 20:15:48 +11:00
|
|
|
|
}
|
2016-01-13 23:12:03 +11:00
|
|
|
|
|
|
|
|
|
static float Make( float value ) { return scale - (scale * value * 2); }
|
|
|
|
|
|
|
|
|
|
internal static int lastTexId, texIndex;
|
|
|
|
|
static void FlushIfNotSame() {
|
|
|
|
|
int texId = atlas.TexIds[texIndex];
|
|
|
|
|
if( texId == lastTexId ) return;
|
|
|
|
|
|
|
|
|
|
if( lastTexId != -1 ) {
|
|
|
|
|
for( int i = 0; i < index; i++ )
|
|
|
|
|
TransformVertex( ref cache.vertices[i] );
|
|
|
|
|
api.UpdateDynamicIndexedVb( DrawMode.Triangles, cache.vb,
|
|
|
|
|
cache.vertices, index, index * 6 / 4 );
|
|
|
|
|
index = 0;
|
|
|
|
|
}
|
|
|
|
|
lastTexId = texId;
|
|
|
|
|
api.BindTexture( texId );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void TransformVertex( ref VertexPos3fTex2fCol4b vertex ) {
|
|
|
|
|
Vector3 p = new Vector3( vertex.X, vertex.Y, vertex.Z );
|
|
|
|
|
//p = Utils.RotateY( p - pos, time ) + pos;
|
|
|
|
|
// See comment in IGraphicsApi.Draw2DTexture()
|
|
|
|
|
p.X -= 0.5f; p.Y -= 0.5f;
|
|
|
|
|
p = Utils.RotateX( Utils.RotateY( p, cosY, sinY ), cosX, sinX );
|
|
|
|
|
vertex.X = p.X; vertex.Y = p.Y; vertex.Z = p.Z;
|
|
|
|
|
}
|
2015-10-06 20:15:48 +11:00
|
|
|
|
}
|
|
|
|
|
}
|