/newcode/src/com/pclewis/mcpatcher/mod/MobRandomizer.java

https://github.com/pclewis/mcpatcher · Java · 213 lines · 190 code · 23 blank · 0 comment · 30 complexity · 2249fe7d47ff596c464cd0a2e2381796 MD5 · raw file

  1. package com.pclewis.mcpatcher.mod;
  2. import com.pclewis.mcpatcher.MCLogger;
  3. import com.pclewis.mcpatcher.MCPatcherUtils;
  4. import com.pclewis.mcpatcher.TexturePackAPI;
  5. import net.minecraft.src.EntityLiving;
  6. import net.minecraft.src.NBTTagCompound;
  7. import java.lang.ref.Reference;
  8. import java.lang.ref.ReferenceQueue;
  9. import java.lang.ref.WeakReference;
  10. import java.lang.reflect.Method;
  11. import java.util.HashMap;
  12. import java.util.HashSet;
  13. public class MobRandomizer {
  14. private static final MCLogger logger = MCLogger.getLogger(MCPatcherUtils.RANDOM_MOBS);
  15. private static final HashMap<EntityLiving, String> cache = new HashMap<EntityLiving, String>();
  16. static {
  17. TexturePackAPI.ChangeHandler.register(new TexturePackAPI.ChangeHandler(MCPatcherUtils.RANDOM_MOBS, 2) {
  18. @Override
  19. protected void onChange() {
  20. cache.clear();
  21. MobRuleList.clear();
  22. MobOverlay.reset();
  23. }
  24. });
  25. }
  26. public static String randomTexture(EntityLiving entity) {
  27. String texture = cache.get(entity);
  28. if (texture == null) {
  29. texture = randomTexture(entity, entity.getEntityTexture());
  30. cache.put(entity, texture);
  31. logger.finer("entity %s using %s", entity, texture);
  32. }
  33. return texture;
  34. }
  35. public static String randomTexture(EntityLiving entity, String texture) {
  36. if (!texture.startsWith("/mob/") || !texture.endsWith(".png")) {
  37. return texture;
  38. }
  39. ExtraInfo info = ExtraInfo.getInfo(entity);
  40. MobRuleList list = MobRuleList.get(texture);
  41. return list.getSkin(info.skin, info.origX, info.origY, info.origZ, info.origBiome);
  42. }
  43. public static String randomTexture(Object entity, String texture) {
  44. if (entity instanceof EntityLiving) {
  45. return randomTexture((EntityLiving) entity, texture);
  46. } else {
  47. return texture;
  48. }
  49. }
  50. public static final class ExtraInfo {
  51. private static final String SKIN_TAG = "randomMobsSkin";
  52. private static final String ORIG_X_TAG = "origX";
  53. private static final String ORIG_Y_TAG = "origY";
  54. private static final String ORIG_Z_TAG = "origZ";
  55. private static final long MULTIPLIER = 0x5deece66dL;
  56. private static final long ADDEND = 0xbL;
  57. private static final long MASK = (1L << 48) - 1;
  58. private static Method getBiomeNameAt;
  59. private static final HashMap<Integer, ExtraInfo> allInfo = new HashMap<Integer, ExtraInfo>();
  60. private static final HashMap<WeakReference<EntityLiving>, ExtraInfo> allRefs = new HashMap<WeakReference<EntityLiving>, ExtraInfo>();
  61. private static final ReferenceQueue<EntityLiving> refQueue = new ReferenceQueue<EntityLiving>();
  62. private final int entityId;
  63. private final HashSet<WeakReference<EntityLiving>> references;
  64. private final long skin;
  65. private final int origX;
  66. private final int origY;
  67. private final int origZ;
  68. private String origBiome;
  69. static {
  70. try {
  71. Class<?> biomeHelperClass = Class.forName(MCPatcherUtils.BIOME_HELPER_CLASS);
  72. getBiomeNameAt = biomeHelperClass.getDeclaredMethod("getBiomeNameAt", Integer.TYPE, Integer.TYPE, Integer.TYPE);
  73. } catch (Throwable e) {
  74. }
  75. if (getBiomeNameAt == null) {
  76. logger.warning("biome integration failed");
  77. } else {
  78. logger.fine("biome integration active");
  79. }
  80. }
  81. ExtraInfo(EntityLiving entity) {
  82. this(entity, getSkinId(entity.entityId), (int) entity.posX, (int) entity.posY, (int) entity.posZ);
  83. }
  84. ExtraInfo(EntityLiving entity, long skin, int origX, int origY, int origZ) {
  85. entityId = entity.entityId;
  86. references = new HashSet<WeakReference<EntityLiving>>();
  87. this.skin = skin;
  88. this.origX = origX;
  89. this.origY = origY;
  90. this.origZ = origZ;
  91. }
  92. private void setBiome() {
  93. if (origBiome == null && getBiomeNameAt != null) {
  94. try {
  95. String biome = (String) getBiomeNameAt.invoke(null, origX, origY, origZ);
  96. if (biome != null) {
  97. origBiome = biome.toLowerCase().replace(" ", "");
  98. }
  99. } catch (Throwable e) {
  100. getBiomeNameAt = null;
  101. e.printStackTrace();
  102. }
  103. }
  104. }
  105. @Override
  106. public String toString() {
  107. return String.format("%s{%d, %d, %d, %d, %d, %s}", getClass().getSimpleName(), entityId, skin, origX, origY, origZ, origBiome);
  108. }
  109. private static void clearUnusedReferences() {
  110. synchronized (allInfo) {
  111. Reference<? extends EntityLiving> ref;
  112. while ((ref = refQueue.poll()) != null) {
  113. ExtraInfo info = allRefs.get(ref);
  114. if (info != null) {
  115. info.references.remove(ref);
  116. if (info.references.isEmpty()) {
  117. logger.finest("removing unused ref %d", info.entityId);
  118. allInfo.remove(info.entityId);
  119. }
  120. }
  121. allRefs.remove(ref);
  122. }
  123. }
  124. }
  125. static ExtraInfo getInfo(EntityLiving entity) {
  126. ExtraInfo info;
  127. synchronized (allInfo) {
  128. clearUnusedReferences();
  129. info = allInfo.get(entity.entityId);
  130. if (info == null) {
  131. info = new ExtraInfo(entity);
  132. putInfo(entity, info);
  133. }
  134. boolean found = false;
  135. for (WeakReference<EntityLiving> ref : info.references) {
  136. if (ref.get() == entity) {
  137. found = true;
  138. break;
  139. }
  140. }
  141. if (!found) {
  142. WeakReference<EntityLiving> reference = new WeakReference<EntityLiving>(entity, refQueue);
  143. info.references.add(reference);
  144. allRefs.put(reference, info);
  145. logger.finest("added ref #%d for %d (%d entities)", info.references.size(), entity.entityId, allInfo.size());
  146. }
  147. info.setBiome();
  148. }
  149. return info;
  150. }
  151. static void putInfo(EntityLiving entity, ExtraInfo info) {
  152. synchronized (allInfo) {
  153. allInfo.put(entity.entityId, info);
  154. }
  155. }
  156. static void clearInfo() {
  157. synchronized (allInfo) {
  158. allInfo.clear();
  159. }
  160. }
  161. private static long getSkinId(int entityId) {
  162. long n = entityId;
  163. n = n ^ (n << 16) ^ (n << 32) ^ (n << 48);
  164. n = MULTIPLIER * n + ADDEND;
  165. n = MULTIPLIER * n + ADDEND;
  166. n &= MASK;
  167. return (n >> 32) ^ n;
  168. }
  169. public static void readFromNBT(EntityLiving entity, NBTTagCompound nbt) {
  170. long skin = nbt.getLong(SKIN_TAG);
  171. if (skin != 0L) {
  172. int x = nbt.getInteger(ORIG_X_TAG);
  173. int y = nbt.getInteger(ORIG_Y_TAG);
  174. int z = nbt.getInteger(ORIG_Z_TAG);
  175. putInfo(entity, new ExtraInfo(entity, skin, x, y, z));
  176. }
  177. }
  178. public static void writeToNBT(EntityLiving entity, NBTTagCompound nbt) {
  179. synchronized (allInfo) {
  180. ExtraInfo info = allInfo.get(entity.entityId);
  181. if (info != null) {
  182. nbt.setLong(SKIN_TAG, info.skin);
  183. nbt.setInteger(ORIG_X_TAG, info.origX);
  184. nbt.setInteger(ORIG_Y_TAG, info.origY);
  185. nbt.setInteger(ORIG_Z_TAG, info.origZ);
  186. }
  187. }
  188. }
  189. }
  190. }