/newcode/src/com/prupe/mcpatcher/mal/tile/FaceInfo.java

https://bitbucket.org/prupe/mcpatcher · Java · 280 lines · 240 code · 38 blank · 2 comment · 25 complexity · e886fdd11caa41d921709da03021595d MD5 · raw file

  1. package com.prupe.mcpatcher.mal.tile;
  2. import com.prupe.mcpatcher.MCLogger;
  3. import net.minecraft.src.*;
  4. import java.util.IdentityHashMap;
  5. import java.util.Map;
  6. final public class FaceInfo {
  7. private static final MCLogger logger = MCLogger.getLogger("Tilesheet");
  8. // duplicated from RenderBlockState
  9. private static final int BOTTOM_FACE = 0; // 0, -1, 0
  10. private static final int TOP_FACE = 1; // 0, 1, 0
  11. private static final int NORTH_FACE = 2; // 0, 0, -1
  12. private static final int SOUTH_FACE = 3; // 0, 0, 1
  13. private static final int WEST_FACE = 4; // -1, 0, 0
  14. private static final int EAST_FACE = 5; // 1, 0, 0
  15. private static final int[] GO_DOWN = new int[]{0, -1, 0};
  16. private static final int[] GO_UP = new int[]{0, 1, 0};
  17. private static final int[] GO_NORTH = new int[]{0, 0, -1};
  18. private static final int[] GO_SOUTH = new int[]{0, 0, 1};
  19. private static final int[] GO_WEST = new int[]{-1, 0, 0};
  20. private static final int[] GO_EAST = new int[]{1, 0, 0};
  21. private static final int[][] NORMALS = new int[][]{
  22. GO_DOWN,
  23. GO_UP,
  24. GO_NORTH,
  25. GO_SOUTH,
  26. GO_WEST,
  27. GO_EAST,
  28. };
  29. private static final Map<ModelFace, FaceInfo> faceInfoMap = new IdentityHashMap<ModelFace, FaceInfo>();
  30. private final ModelFace face;
  31. private final TextureAtlasSprite sprite;
  32. private final String textureName;
  33. private final int effectiveFace;
  34. private final int uvRotation;
  35. private final Map<Icon, ModelFace> altIcons = new IdentityHashMap<Icon, ModelFace>();
  36. private ModelFace nonAtlasFace;
  37. static void clear() {
  38. faceInfoMap.clear();
  39. }
  40. public static ModelFace registerModelFaceSprite(ModelFace face, TextureAtlasSprite sprite, String textureName) {
  41. FaceInfo faceInfo = faceInfoMap.get(face);
  42. if (faceInfo == null) {
  43. faceInfo = new FaceInfo(face, sprite, textureName);
  44. faceInfoMap.put(face, faceInfo);
  45. }
  46. return face;
  47. }
  48. public static void registerModelFaceSprite(ModelFace face, TextureAtlasSprite sprite) {
  49. registerModelFaceSprite(face, sprite, IconAPI.getIconName(sprite));
  50. }
  51. public static FaceInfo getFaceInfo(ModelFace face) {
  52. return faceInfoMap.get(face);
  53. }
  54. private FaceInfo(ModelFace face, TextureAtlasSprite sprite, String textureName) {
  55. this.face = face;
  56. if (face instanceof ModelFaceSprite) {
  57. this.sprite = ((ModelFaceSprite) face).sprite;
  58. } else {
  59. this.sprite = sprite;
  60. }
  61. if (textureName.startsWith("#")) {
  62. textureName = textureName.substring(1);
  63. }
  64. textureName = textureName.toLowerCase().trim();
  65. this.textureName = textureName;
  66. effectiveFace = computeEffectiveFace(face.getIntBuffer());
  67. uvRotation = computeRotation(face.getIntBuffer(), getEffectiveFace());
  68. }
  69. @Override
  70. public String toString() {
  71. return String.format("FaceInfo{%s,%s#%s -> %s R%+d}",
  72. face, sprite.getIconName(), textureName, Direction.values()[effectiveFace], uvRotation * 45
  73. );
  74. }
  75. public TextureAtlasSprite getSprite() {
  76. return sprite;
  77. }
  78. public ModelFace getAltFace(TextureAtlasSprite altSprite) {
  79. if (altSprite == sprite || altSprite == null) {
  80. return face;
  81. }
  82. synchronized (this) {
  83. ModelFace newFace = altIcons.get(altSprite);
  84. if (newFace == null) {
  85. newFace = new ModelFaceSprite(face, altSprite);
  86. calculateSpriteUV(sprite, face.getIntBuffer(), altSprite, newFace.getIntBuffer());
  87. altIcons.put(altSprite, newFace);
  88. }
  89. return newFace;
  90. }
  91. }
  92. // NOTE: not synchronized as item rendering is single-threaded
  93. public ModelFace getNonAtlasFace() {
  94. if (nonAtlasFace == null) {
  95. nonAtlasFace = new ModelFaceSprite(face, sprite);
  96. calculateNonAtlasUV(sprite, face.getIntBuffer(), nonAtlasFace.getIntBuffer());
  97. }
  98. return nonAtlasFace;
  99. }
  100. public int getEffectiveFace() {
  101. return effectiveFace;
  102. }
  103. public int getUVRotation() {
  104. return uvRotation;
  105. }
  106. public String getTextureName() {
  107. return textureName;
  108. }
  109. private static int computeEffectiveFace(int[] b) {
  110. float ax = Float.intBitsToFloat(b[0]) - Float.intBitsToFloat(b[7]); // x0 - x1
  111. float ay = Float.intBitsToFloat(b[1]) - Float.intBitsToFloat(b[8]); // y0 - y1
  112. float az = Float.intBitsToFloat(b[2]) - Float.intBitsToFloat(b[9]); // z0 - z1
  113. float bx = Float.intBitsToFloat(b[0]) - Float.intBitsToFloat(b[21]); // x0 - x3
  114. float by = Float.intBitsToFloat(b[1]) - Float.intBitsToFloat(b[22]); // y0 - y3
  115. float bz = Float.intBitsToFloat(b[2]) - Float.intBitsToFloat(b[23]); // z0 - z3
  116. float cx = ay * bz - by * az; // cross product
  117. float cy = bx * az - ax * bz;
  118. float cz = ax * by - bx * ay;
  119. int bigidx = -1;
  120. float[] dot = new float[6];
  121. float bigdot = 0.0f;
  122. for (int i = 0; i < NORMALS.length; i++) {
  123. int[] n = NORMALS[i];
  124. dot[i] = (float) n[0] * cx + (float) n[1] * cy + (float) n[2] * cz;
  125. if (dot[i] > bigdot) {
  126. bigdot = dot[i];
  127. bigidx = i;
  128. }
  129. }
  130. if (bigidx < 0) {
  131. logger.warning("a: %f %f %f", ax, ay, az);
  132. logger.warning("b: %f %f %f", bx, by, bz);
  133. logger.warning("c: %f %f %f", cx, cy, cz);
  134. logger.warning("dot: %f %f %f %f %f %f", dot[0], dot[1], dot[2], dot[3], dot[4], dot[5]);
  135. bigidx = NORTH_FACE;
  136. }
  137. return bigidx;
  138. }
  139. private static int computeRotation(int[] b, int face) {
  140. return (computeXYZRotation(b, face) - computeUVRotation(b)) & 0x7;
  141. }
  142. private static int computeUVRotation(int[] b) {
  143. float du = Float.intBitsToFloat(b[4]) - Float.intBitsToFloat(b[18]); // u0 - u2
  144. float dv = Float.intBitsToFloat(b[5]) - Float.intBitsToFloat(b[19]); // v0 - v2
  145. return computeRotation(du, dv);
  146. }
  147. private static int computeXYZRotation(int[] b, int face) {
  148. float dx = Float.intBitsToFloat(b[0]) - Float.intBitsToFloat(b[14]); // x0 - x2
  149. float dy = Float.intBitsToFloat(b[1]) - Float.intBitsToFloat(b[15]); // y0 - y2
  150. float dz = Float.intBitsToFloat(b[2]) - Float.intBitsToFloat(b[16]); // z0 - z2
  151. float du;
  152. float dv;
  153. switch (face) {
  154. case BOTTOM_FACE:
  155. du = dx;
  156. dv = -dz;
  157. break;
  158. case TOP_FACE:
  159. du = dx;
  160. dv = dz;
  161. break;
  162. case NORTH_FACE:
  163. du = -dx;
  164. dv = -dy;
  165. break;
  166. case SOUTH_FACE:
  167. du = dx;
  168. dv = -dy;
  169. break;
  170. case WEST_FACE:
  171. du = dz;
  172. dv = -dy;
  173. break;
  174. case EAST_FACE:
  175. du = -dz;
  176. dv = -dy;
  177. break;
  178. default:
  179. return 0;
  180. }
  181. return computeRotation(du, dv);
  182. }
  183. private static int computeRotation(float s, float t) {
  184. if (s <= 0) {
  185. if (t <= 0) {
  186. return 0; // no rotation
  187. } else {
  188. return 2; // rotate 90 ccw
  189. }
  190. } else {
  191. if (t <= 0) {
  192. return 6; // rotate 90 cw
  193. } else {
  194. return 4; // rotate 180
  195. }
  196. }
  197. }
  198. private static void logRotation(int[] b, int face) {
  199. int x0 = (int) (Float.intBitsToFloat(b[0]) * 16.0f);
  200. int y0 = (int) (Float.intBitsToFloat(b[1]) * 16.0f);
  201. int z0 = (int) (Float.intBitsToFloat(b[2]) * 16.0f);
  202. int u0 = (int) Float.intBitsToFloat(b[4]);
  203. int v0 = (int) Float.intBitsToFloat(b[5]);
  204. int x1 = (int) (Float.intBitsToFloat(b[7]) * 16.0f);
  205. int y1 = (int) (Float.intBitsToFloat(b[8]) * 16.0f);
  206. int z1 = (int) (Float.intBitsToFloat(b[9]) * 16.0f);
  207. int u1 = (int) Float.intBitsToFloat(b[11]);
  208. int v1 = (int) Float.intBitsToFloat(b[12]);
  209. int x2 = (int) (Float.intBitsToFloat(b[14]) * 16.0f);
  210. int y2 = (int) (Float.intBitsToFloat(b[15]) * 16.0f);
  211. int z2 = (int) (Float.intBitsToFloat(b[16]) * 16.0f);
  212. int u2 = (int) Float.intBitsToFloat(b[18]);
  213. int v2 = (int) Float.intBitsToFloat(b[19]);
  214. int x3 = (int) (Float.intBitsToFloat(b[21]) * 16.0f);
  215. int y3 = (int) (Float.intBitsToFloat(b[22]) * 16.0f);
  216. int z3 = (int) (Float.intBitsToFloat(b[23]) * 16.0f);
  217. int u3 = (int) Float.intBitsToFloat(b[25]);
  218. int v3 = (int) Float.intBitsToFloat(b[26]);
  219. logger.info("x0,y0,z0=%d %d %d, x1,y1,z1=%d %d %d, x2,y2,z2=%d %d %d, x3,y3,z3=%d %d %d, rotation %d",
  220. x0, y0, z0, x1, y1, z1, x2, y2, z2, x3, y3, z3, computeXYZRotation(b, face)
  221. );
  222. logger.info("u0,v0=%d %d, u1,v1=%d %d, u2,v2=%d %d, u3,v3=%d %d, rotation %d",
  223. u0, v0, u1, v1, u2, v2, u3, v3, computeUVRotation(b)
  224. );
  225. }
  226. private static void calculateSpriteUV(TextureAtlasSprite origIcon, int[] a, TextureAtlasSprite newIcon, int[] b) {
  227. for (int i = 0; i < 28; i += 7) {
  228. float u = 16.0f * (Float.intBitsToFloat(a[i + 4]) - origIcon.getMinU()) / (origIcon.getMaxU() - origIcon.getMinU());
  229. float v = 16.0f * (Float.intBitsToFloat(a[i + 5]) - origIcon.getMinV()) / (origIcon.getMaxV() - origIcon.getMinV());
  230. b[i + 4] = Float.floatToIntBits(newIcon.getInterpolatedU(u));
  231. b[i + 5] = Float.floatToIntBits(newIcon.getInterpolatedV(v));
  232. }
  233. }
  234. private static void calculateNonAtlasUV(TextureAtlasSprite origIcon, int[] a, int[] b) {
  235. for (int i = 0; i < 28; i += 7) {
  236. float u = (Float.intBitsToFloat(a[i + 4]) - origIcon.getMinU()) / (origIcon.getMaxU() - origIcon.getMinU());
  237. float v = (Float.intBitsToFloat(a[i + 5]) - origIcon.getMinV()) / (origIcon.getMaxV() - origIcon.getMinV());
  238. b[i + 4] = Float.floatToIntBits(u);
  239. b[i + 5] = Float.floatToIntBits(v);
  240. }
  241. }
  242. }