diff --git a/src/main/java/org/dynmap/hdmap/IsoHDPerspective.java b/src/main/java/org/dynmap/hdmap/IsoHDPerspective.java index 92f342b6..57562d83 100644 --- a/src/main/java/org/dynmap/hdmap/IsoHDPerspective.java +++ b/src/main/java/org/dynmap/hdmap/IsoHDPerspective.java @@ -96,6 +96,8 @@ public class IsoHDPerspective implements HDPerspective { Vector3D top, bottom; int px, py; BlockStep laststep = BlockStep.Y_MINUS; + + BlockStep stepx, stepy, stepz; /* Raytrace state variables */ double dx, dy, dz; int x, y, z; @@ -115,6 +117,7 @@ public class IsoHDPerspective implements HDPerspective { int subalpha; double mt; double mtend; + int mxout, myout, mzout; int[] subblock_xyz = new int[3]; MapIterator mapiter; @@ -238,56 +241,76 @@ public class IsoHDPerspective implements HDPerspective { /* Compute number of steps and increments for each */ n = 1; + int mxout, myout, mzout; + /* If perpendicular to X axis */ if (dx == 0) { x_inc = 0; t_next_x = Double.MAX_VALUE; + stepx = BlockStep.X_PLUS; + mxout = modscale; } /* If bottom is right of top */ else if (bottom.x > top.x) { x_inc = 1; n += (int) (Math.floor(bottom.x)) - x; t_next_x = (Math.floor(top.x) + 1 - top.x) * dt_dx; + stepx = BlockStep.X_PLUS; + mxout = modscale; } /* Top is right of bottom */ else { x_inc = -1; n += x - (int) (Math.floor(bottom.x)); t_next_x = (top.x - Math.floor(top.x)) * dt_dx; + stepx = BlockStep.X_MINUS; + mxout = -1; } /* If perpendicular to Y axis */ if (dy == 0) { y_inc = 0; t_next_y = Double.MAX_VALUE; + stepy = BlockStep.Y_PLUS; + myout = modscale; } /* If bottom is above top */ else if (bottom.y > top.y) { y_inc = 1; n += (int) (Math.floor(bottom.y)) - y; t_next_y = (Math.floor(top.y) + 1 - top.y) * dt_dy; + stepy = BlockStep.Y_PLUS; + myout = modscale; } /* If top is above bottom */ else { y_inc = -1; n += y - (int) (Math.floor(bottom.y)); t_next_y = (top.y - Math.floor(top.y)) * dt_dy; + stepy = BlockStep.Y_MINUS; + myout = -1; } /* If perpendicular to Z axis */ if (dz == 0) { z_inc = 0; t_next_z = Double.MAX_VALUE; + stepz = BlockStep.Z_PLUS; + mzout = modscale; } /* If bottom right of top */ else if (bottom.z > top.z) { z_inc = 1; n += (int) (Math.floor(bottom.z)) - z; t_next_z = (Math.floor(top.z) + 1 - top.z) * dt_dz; + stepz = BlockStep.Z_PLUS; + mzout = modscale; } /* If bottom left of top */ else { z_inc = -1; n += z - (int) (Math.floor(bottom.z)); t_next_z = (top.z - Math.floor(top.z)) * dt_dz; + stepz = BlockStep.Z_MINUS; + mzout = -1; } /* Walk through scene */ laststep = BlockStep.Y_MINUS; /* Last step is down into map */ @@ -363,10 +386,29 @@ public class IsoHDPerspective implements HDPerspective { return cd.ordinal(); } + private final boolean handleSubModel(short[] model, HDShaderState[] shaderstate, boolean[] shaderdone) { + boolean firststep = true; + + while(!raytraceSubblock(model, firststep)) { + boolean done = true; + skylevel = emitlevel = -1; + for(int i = 0; i < shaderstate.length; i++) { + if(!shaderdone[i]) + shaderdone[i] = shaderstate[i].processBlock(this); + done = done && shaderdone[i]; + } + /* If all are done, we're out */ + if(done) + return true; + nonairhit = true; + firststep = false; + } + return false; + } /** * Process visit of ray to block */ - private boolean visit_block(MapIterator mapiter, HDShaderState[] shaderstate, boolean[] shaderdone) { + private final boolean visit_block(MapIterator mapiter, HDShaderState[] shaderstate, boolean[] shaderdone) { lastblocktypeid = blocktypeid; blocktypeid = mapiter.getBlockTypeID(); if(skiptoair) { /* If skipping until we see air */ @@ -374,33 +416,21 @@ public class IsoHDPerspective implements HDPerspective { skiptoair = false; } else if(nonairhit || (blocktypeid != 0)) { - blockdata = mapiter.getBlockData(); - if(blocktypeid == FENCE_BLKTYPEID) { /* Special case for fence - need to fake data so we can render properly */ - blockdata = generateFenceBlockData(mapiter); + switch(blocktypeid) { + case FENCE_BLKTYPEID: /* Special case for fence - need to fake data so we can render properly */ + blockdata = generateFenceBlockData(mapiter); + break; + case CHEST_BLKTYPEID: /* Special case for chest - need to fake data so we can render */ + blockdata = generateChestBlockData(mapiter); + break; + default: + blockdata = mapiter.getBlockData(); + break; } - else if(blocktypeid == CHEST_BLKTYPEID) { /* Special case for chest - need to fake data so we can render */ - blockdata = generateChestBlockData(mapiter); - } - /* Look up to see if block is modelled */ short[] model = scalemodels.getScaledModel(blocktypeid, blockdata); if(model != null) { - boolean firststep = true; - - while(!raytraceSubblock(model, firststep)) { - boolean done = true; - skylevel = emitlevel = -1; - for(int i = 0; i < shaderstate.length; i++) { - if(!shaderdone[i]) - shaderdone[i] = shaderstate[i].processBlock(this); - done = done && shaderdone[i]; - } - /* If all are done, we're out */ - if(done) - return true; - nonairhit = true; - firststep = false; - } + return handleSubModel(model, shaderstate, shaderdone); } else { boolean done = true; @@ -437,12 +467,7 @@ public class IsoHDPerspective implements HDPerspective { x += x_inc; t = t_next_x; t_next_x += dt_dx; - if(x_inc > 0) { - laststep = BlockStep.X_PLUS; - } - else { - laststep = BlockStep.X_MINUS; - } + laststep = stepx; mapiter.stepPosition(laststep); } /* If Y step is next best */ @@ -450,30 +475,17 @@ public class IsoHDPerspective implements HDPerspective { y += y_inc; t = t_next_y; t_next_y += dt_dy; - if(y_inc > 0) { - laststep = BlockStep.Y_PLUS; - mapiter.stepPosition(laststep); - if(mapiter.getY() > 127) - return; - } - else { - laststep = BlockStep.Y_MINUS; - mapiter.stepPosition(laststep); - if(mapiter.getY() < 0) - return; - } + laststep = stepy; + mapiter.stepPosition(laststep); + /* If outside 0-127 range */ + if((y & (~0x7F)) != 0) return; } /* Else, Z step is next best */ else { z += z_inc; t = t_next_z; t_next_z += dt_dz; - if(z_inc > 0) { - laststep = BlockStep.Z_PLUS; - } - else { - laststep = BlockStep.Z_MINUS; - } + laststep = stepz; mapiter.stepPosition(laststep); } } @@ -531,56 +543,32 @@ public class IsoHDPerspective implements HDPerspective { mx += x_inc; mt = mt_next_x; mt_next_x += mdt_dx; - if(x_inc > 0) { - laststep = BlockStep.X_PLUS; - if(mx >= modscale) - return true; - } - else { - laststep = BlockStep.X_MINUS; - if(mx < 0) - //mx += modscale; - return true; - } + laststep = stepx; + if(mx == mxout) + return true; } /* If Y step is next best */ else if((mt_next_y <= mt_next_x) && (mt_next_y <= mt_next_z)) { my += y_inc; mt = mt_next_y; mt_next_y += mdt_dy; - if(y_inc > 0) { - laststep = BlockStep.Y_PLUS; - if(my >= modscale) - return true; - } - else { - laststep = BlockStep.Y_MINUS; - if(my < 0) - //my += modscale; - return true; - } + laststep = stepy; + if(my == myout) + return true; } /* Else, Z step is next best */ else { mz += z_inc; mt = mt_next_z; mt_next_z += mdt_dz; - if(z_inc > 0) { - laststep = BlockStep.Z_PLUS; - if(mz >= modscale) - return true; - } - else { - laststep = BlockStep.Z_MINUS; - if(mz < 0) - //mz += modscale; - return true; - } + laststep = stepz; + if(mz == mzout) + return true; } } return true; } - public int[] getSubblockCoord() { + public final int[] getSubblockCoord() { double tt = t + 0.000001; if(subalpha >= 0) tt = mt; diff --git a/src/main/java/org/dynmap/hdmap/TexturePack.java b/src/main/java/org/dynmap/hdmap/TexturePack.java index 367ccda0..42c8dd86 100644 --- a/src/main/java/org/dynmap/hdmap/TexturePack.java +++ b/src/main/java/org/dynmap/hdmap/TexturePack.java @@ -55,9 +55,10 @@ public class TexturePack { private static final String CUSTOMWATERFLOWING_PNG = "custom_water_flowing.png"; /* Color modifier codes (x1000 for value in mapping code) */ + private static final int COLORMOD_NONE = 0; private static final int COLORMOD_GRASSTONED = 1; private static final int COLORMOD_FOLIAGETONED = 2; -// private static final int COLORMOD_WATERTONED = 3; + private static final int COLORMOD_WATERTONED = 3; /* Not used */ private static final int COLORMOD_ROT90 = 4; private static final int COLORMOD_ROT180 = 5; private static final int COLORMOD_ROT270 = 6; @@ -126,7 +127,7 @@ public class TexturePack { private HDTextureMap() { blockids = Collections.singletonList(Integer.valueOf(0)); databits = 0xFFFF; - faces = new int[] { -1, -1, -1, -1, -1, -1 }; + faces = new int[] { BLOCKINDEX_BLANK, BLOCKINDEX_BLANK, BLOCKINDEX_BLANK, BLOCKINDEX_BLANK, BLOCKINDEX_BLANK, BLOCKINDEX_BLANK }; for(int i = 0; i < texmaps.length; i++) { texmaps[i] = this; @@ -703,7 +704,7 @@ public class TexturePack { /** * Read color for given subblock coordinate, with given block id and data and face */ - public void readColor(HDPerspectiveState ps, MapIterator mapiter, Color rslt, int blkid, int lastblocktype, boolean biome_shaded) { + public final void readColor(final HDPerspectiveState ps, final MapIterator mapiter, final Color rslt, final int blkid, final int lastblocktype, final boolean biome_shaded) { int blkdata = ps.getBlockData(); HDTextureMap map = HDTextureMap.getMap(blkid, blkdata); BlockStep laststep = ps.getLastBlockStep(); @@ -712,6 +713,35 @@ public class TexturePack { rslt.setTransparent(); return; } + else if(textid < 1000) { /* If simple mapping */ + int[] texture = terrain_argb[textid]; + int[] xyz = ps.getSubblockCoord(); + /* Get texture coordinates (U=horizontal(left=0),V=vertical(top=0)) */ + int u = 0, v = 0; + + switch(laststep) { + case X_MINUS: /* South face: U = East (Z-), V = Down (Y-) */ + u = native_scale-xyz[2]-1; v = native_scale-xyz[1]-1; + break; + case X_PLUS: /* North face: U = West (Z+), V = Down (Y-) */ + u = xyz[2]; v = native_scale-xyz[1]-1; + break; + case Z_MINUS: /* West face: U = South (X+), V = Down (Y-) */ + u = xyz[0]; v = native_scale-xyz[1]-1; + break; + case Z_PLUS: /* East face: U = North (X-), V = Down (Y-) */ + u = native_scale-xyz[0]-1; v = native_scale-xyz[1]-1; + break; + case Y_MINUS: /* U = East(Z-), V = South(X+) */ + case Y_PLUS: + u = native_scale-xyz[2]-1; v = xyz[0]; + break; + } + /* Read color from texture */ + rslt.setARGB(texture[v*native_scale + u]); + return; + } + /* See if not basic block texture */ int textop = textid / 1000; textid = textid % 1000; @@ -749,80 +779,89 @@ public class TexturePack { break; } /* Handle U-V transorms before fetching color */ - if(textop > 0) { - switch(textop) { - case COLORMOD_ROT90: - tmp = u; u = native_scale - v - 1; v = tmp; - break; - case COLORMOD_ROT180: - u = native_scale - u - 1; v = native_scale - v - 1; - break; - case COLORMOD_ROT270: - tmp = u; u = v; v = native_scale - tmp - 1; - break; - case COLORMOD_FLIPHORIZ: - u = native_scale - u - 1; - break; - case COLORMOD_SHIFTDOWNHALF: - case COLORMOD_SHIFTDOWNHALFANDFLIPHORIZ: - if(v < native_scale/2) { - rslt.setTransparent(); - return; + switch(textop) { + case COLORMOD_NONE: + case COLORMOD_GRASSTONED: + case COLORMOD_FOLIAGETONED: + case COLORMOD_WATERTONED: + break; + case COLORMOD_ROT90: + tmp = u; u = native_scale - v - 1; v = tmp; + break; + case COLORMOD_ROT180: + u = native_scale - u - 1; v = native_scale - v - 1; + break; + case COLORMOD_ROT270: + tmp = u; u = v; v = native_scale - tmp - 1; + break; + case COLORMOD_FLIPHORIZ: + u = native_scale - u - 1; + break; + case COLORMOD_SHIFTDOWNHALF: + if(v < native_scale/2) { + rslt.setTransparent(); + return; + } + v -= native_scale/2; + break; + case COLORMOD_SHIFTDOWNHALFANDFLIPHORIZ: + if(v < native_scale/2) { + rslt.setTransparent(); + return; + } + v -= native_scale/2; + u = native_scale - u - 1; + break; + case COLORMOD_INCLINEDTORCH: + if(v >= (3*native_scale/4)) { + rslt.setTransparent(); + return; + } + v += native_scale/4; + if(u < native_scale/2) u = native_scale/2-1; + if(u > native_scale/2) u = native_scale/2; + break; + case COLORMOD_GRASSSIDE: + /* Check if snow above block */ + if(mapiter.getBlockTypeIDAt(BlockStep.Y_PLUS) == 78) { + texture = terrain_argb[68]; /* Snow block */ + textid = 68; + } + else { /* Else, check the grass color overlay */ + int ovclr = terrain_argb[38][v*native_scale+u]; + if((ovclr & 0xFF000000) != 0) { /* Hit? */ + texture = terrain_argb[38]; /* Use it */ + textop = COLORMOD_GRASSTONED; /* Force grass toning */ } - v -= native_scale/2; - if(textop == COLORMOD_SHIFTDOWNHALFANDFLIPHORIZ) - u = native_scale - u - 1; - break; - case COLORMOD_INCLINEDTORCH: - if(v >= (3*native_scale/4)) { - rslt.setTransparent(); - return; - } - v += native_scale/4; - if(u < native_scale/2) u = native_scale/2-1; - if(u > native_scale/2) u = native_scale/2; - break; - case COLORMOD_GRASSSIDE: - /* Check if snow above block */ - if(mapiter.getBlockTypeIDAt(BlockStep.Y_PLUS) == 78) { - texture = terrain_argb[68]; /* Snow block */ - textid = 68; - } - else { /* Else, check the grass color overlay */ - int ovclr = terrain_argb[38][v*native_scale+u]; - if((ovclr & 0xFF000000) != 0) { /* Hit? */ - texture = terrain_argb[38]; /* Use it */ - textop = COLORMOD_GRASSTONED; /* Force grass toning */ - } - } - break; - } + } + break; + case COLORMOD_CLEARINSIDE: + break; } /* Read color from texture */ rslt.setARGB(texture[v*native_scale + u]); - if(textop > 0) { - LoadedImage li; - /* Switch based on texture modifier */ - switch(textop) { - case COLORMOD_GRASSTONED: - li = imgs[IMG_GRASSCOLOR]; - if((li.argb == null) || (!biome_shaded)) { - rslt.blendColor(li.trivial_color); - } - else { - rslt.blendColor(biomeLookup(li.argb, li.width, mapiter.getRawBiomeRainfall(), mapiter.getRawBiomeTemperature())); - } - break; - case COLORMOD_FOLIAGETONED: - li = imgs[IMG_FOLIAGECOLOR]; - if((li.argb == null) || (!biome_shaded)) { - rslt.blendColor(li.trivial_color); - } - else { - rslt.blendColor(biomeLookup(li.argb, li.width, mapiter.getRawBiomeRainfall(), mapiter.getRawBiomeTemperature())); - } - break; - } + + LoadedImage li; + /* Switch based on texture modifier */ + switch(textop) { + case COLORMOD_GRASSTONED: + li = imgs[IMG_GRASSCOLOR]; + if((li.argb == null) || (!biome_shaded)) { + rslt.blendColor(li.trivial_color); + } + else { + rslt.blendColor(biomeLookup(li.argb, li.width, mapiter.getRawBiomeRainfall(), mapiter.getRawBiomeTemperature())); + } + break; + case COLORMOD_FOLIAGETONED: + li = imgs[IMG_FOLIAGECOLOR]; + if((li.argb == null) || (!biome_shaded)) { + rslt.blendColor(li.trivial_color); + } + else { + rslt.blendColor(biomeLookup(li.argb, li.width, mapiter.getRawBiomeRainfall(), mapiter.getRawBiomeTemperature())); + } + break; } }