/newcode/src/com/prupe/mcpatcher/cc/ColorMapBase.java

https://bitbucket.org/SevenBits/mcpatcher · Java · 435 lines · 370 code · 65 blank · 0 comment · 40 complexity · 354eedb44502c1c82fda6d721463957b MD5 · raw file

  1. package com.prupe.mcpatcher.cc;
  2. import com.prupe.mcpatcher.MCLogger;
  3. import com.prupe.mcpatcher.MCPatcherUtils;
  4. import net.minecraft.src.IBlockAccess;
  5. import net.minecraft.src.ResourceLocation;
  6. import java.util.ArrayList;
  7. import java.util.Arrays;
  8. import java.util.Collection;
  9. import java.util.List;
  10. import java.util.logging.Level;
  11. abstract class ColorMapBase {
  12. private static final MCLogger logger = MCLogger.getLogger(MCPatcherUtils.CUSTOM_COLORS);
  13. static final int DEFAULT_HEIGHT = 64;
  14. static final class Blended implements IColorMap {
  15. private final IColorMap parent;
  16. private final int blendRadius;
  17. private final int[][] offset;
  18. private final float[] weight;
  19. private final float[] lastColor = new float[3];
  20. Blended(IColorMap parent, int blendRadius) {
  21. this.parent = parent;
  22. this.blendRadius = blendRadius;
  23. List<int[]> blendOffset = new ArrayList<int[]>();
  24. List<Float> blendWeight = new ArrayList<Float>();
  25. float blendScale = 0.0f;
  26. for (int r = 0; r <= blendRadius; r++) {
  27. if (r == 0) {
  28. blendScale += addSample(blendOffset, blendWeight, 0, 0);
  29. } else {
  30. switch (r % 8) {
  31. case 1:
  32. blendScale += addSamples(blendOffset, blendWeight, r, 0, 1);
  33. break;
  34. case 2:
  35. blendScale += addSamples(blendOffset, blendWeight, r, 1, 1);
  36. break;
  37. case 3:
  38. case 4:
  39. blendScale += addSamples(blendOffset, blendWeight, r, 1, 2);
  40. break;
  41. case 5:
  42. case 6:
  43. blendScale += addSamples(blendOffset, blendWeight, r, 1, 3);
  44. break;
  45. case 7:
  46. default:
  47. blendScale += addSamples(blendOffset, blendWeight, r, 2, 3);
  48. break;
  49. }
  50. }
  51. }
  52. offset = blendOffset.toArray(new int[blendOffset.size()][]);
  53. this.weight = new float[blendWeight.size()];
  54. for (int i = 0; i < blendWeight.size(); i++) {
  55. this.weight[i] = blendWeight.get(i) / blendScale;
  56. }
  57. }
  58. private static float addSample(List<int[]> blendOffset, List<Float> blendWeight, int di, int dk) {
  59. float weight = (float) Math.pow(1.0f + Math.max(Math.abs(di), Math.abs(dk)), -0.5);
  60. blendOffset.add(new int[]{di, dk});
  61. blendWeight.add(weight);
  62. return weight;
  63. }
  64. private static float addSamples(List<int[]> blendOffset, List<Float> blendWeight, int r, int num, int denom) {
  65. int s = num * r / denom;
  66. float sum = 0.0f;
  67. if (r % 2 == 0) {
  68. r ^= s;
  69. s ^= r;
  70. r ^= s;
  71. }
  72. sum += addSample(blendOffset, blendWeight, r, s);
  73. sum += addSample(blendOffset, blendWeight, -s, r);
  74. sum += addSample(blendOffset, blendWeight, -r, -s);
  75. sum += addSample(blendOffset, blendWeight, s, -r);
  76. return sum;
  77. }
  78. @Override
  79. public String toString() {
  80. return parent.toString();
  81. }
  82. @Override
  83. public boolean isHeightDependent() {
  84. return parent.isHeightDependent();
  85. }
  86. @Override
  87. public int getColorMultiplier() {
  88. return parent.getColorMultiplier();
  89. }
  90. @Override
  91. public int getColorMultiplier(IBlockAccess blockAccess, int i, int j, int k) {
  92. return Colorizer.float3ToInt(getColorMultiplierF(blockAccess, i, j, k));
  93. }
  94. @Override
  95. public float[] getColorMultiplierF(IBlockAccess blockAccess, int i, int j, int k) {
  96. lastColor[0] = 0.0f;
  97. lastColor[1] = 0.0f;
  98. lastColor[2] = 0.0f;
  99. for (int n = 0; n < weight.length; n++) {
  100. int[] offset = this.offset[n];
  101. float weight = this.weight[n];
  102. float[] tmpColor = parent.getColorMultiplierF(blockAccess, i + offset[0], j, k + offset[1]);
  103. lastColor[0] += tmpColor[0] * weight;
  104. lastColor[1] += tmpColor[1] * weight;
  105. lastColor[2] += tmpColor[2] * weight;
  106. }
  107. return lastColor;
  108. }
  109. @Override
  110. public void claimResources(Collection<ResourceLocation> resources) {
  111. parent.claimResources(resources);
  112. }
  113. @Override
  114. public IColorMap copy() {
  115. return new Blended(parent.copy(), blendRadius);
  116. }
  117. }
  118. static final class Cached implements IColorMap {
  119. private final IColorMap parent;
  120. private int lastI = Integer.MIN_VALUE;
  121. private int lastJ = Integer.MIN_VALUE;
  122. private int lastK = Integer.MIN_VALUE;
  123. private int lastColorI;
  124. private float[] lastColorF;
  125. Cached(IColorMap parent) {
  126. this.parent = parent;
  127. }
  128. @Override
  129. public String toString() {
  130. return parent.toString();
  131. }
  132. @Override
  133. public boolean isHeightDependent() {
  134. return parent.isHeightDependent();
  135. }
  136. @Override
  137. public int getColorMultiplier() {
  138. return parent.getColorMultiplier();
  139. }
  140. @Override
  141. public int getColorMultiplier(IBlockAccess blockAccess, int i, int j, int k) {
  142. if (i != lastI || j != lastJ || k != lastK) {
  143. lastColorI = parent.getColorMultiplier(blockAccess, i, j, k);
  144. lastI = i;
  145. lastJ = j;
  146. lastK = k;
  147. }
  148. return lastColorI;
  149. }
  150. @Override
  151. public float[] getColorMultiplierF(IBlockAccess blockAccess, int i, int j, int k) {
  152. if (i != lastI || j != lastJ || k != lastK) {
  153. lastColorF = parent.getColorMultiplierF(blockAccess, i, j, k);
  154. lastI = i;
  155. lastJ = j;
  156. lastK = k;
  157. }
  158. return lastColorF;
  159. }
  160. @Override
  161. public void claimResources(Collection<ResourceLocation> resources) {
  162. parent.claimResources(resources);
  163. }
  164. @Override
  165. public IColorMap copy() {
  166. return new Cached(parent.copy());
  167. }
  168. }
  169. static final class Smoothed implements IColorMap {
  170. private final IColorMap parent;
  171. private final float smoothTime;
  172. private final float[] lastColor = new float[3];
  173. private long lastTime;
  174. Smoothed(IColorMap parent, float smoothTime) {
  175. this.parent = parent;
  176. this.smoothTime = smoothTime;
  177. }
  178. @Override
  179. public String toString() {
  180. return parent.toString();
  181. }
  182. @Override
  183. public boolean isHeightDependent() {
  184. return parent.isHeightDependent();
  185. }
  186. @Override
  187. public int getColorMultiplier() {
  188. return parent.getColorMultiplier();
  189. }
  190. @Override
  191. public int getColorMultiplier(IBlockAccess blockAccess, int i, int j, int k) {
  192. return Colorizer.float3ToInt(getColorMultiplierF(blockAccess, i, j, k));
  193. }
  194. @Override
  195. public float[] getColorMultiplierF(IBlockAccess blockAccess, int i, int j, int k) {
  196. float[] currentColor = parent.getColorMultiplierF(blockAccess, i, j, k);
  197. long now = System.currentTimeMillis();
  198. if (lastTime == 0L) {
  199. lastColor[0] = currentColor[0];
  200. lastColor[1] = currentColor[1];
  201. lastColor[2] = currentColor[2];
  202. } else {
  203. float r = Colorizer.clamp((float) (now - lastTime) / smoothTime);
  204. float s = 1.0f - r;
  205. lastColor[0] = r * currentColor[0] + s * lastColor[0];
  206. lastColor[1] = r * currentColor[1] + s * lastColor[1];
  207. lastColor[2] = r * currentColor[2] + s * lastColor[2];
  208. }
  209. lastTime = now;
  210. return lastColor;
  211. }
  212. @Override
  213. public void claimResources(Collection<ResourceLocation> resources) {
  214. parent.claimResources(resources);
  215. }
  216. @Override
  217. public IColorMap copy() {
  218. return new Smoothed(parent.copy(), smoothTime);
  219. }
  220. }
  221. static final class Chunked implements IColorMap {
  222. private static final int I_MASK = ~0xf;
  223. private static final int K_MASK = ~0xf;
  224. private static final int I_SIZE = 17;
  225. private static final int J_SIZE = 2;
  226. private static final int K_SIZE = 17;
  227. private static final int IK_SIZE = I_SIZE * K_SIZE;
  228. private static final int IJK_SIZE = I_SIZE * J_SIZE * K_SIZE;
  229. private final IColorMap parent;
  230. private int baseI = Integer.MIN_VALUE;
  231. private int baseJ = Integer.MIN_VALUE;
  232. private int baseK = Integer.MIN_VALUE;
  233. private int baseOffset;
  234. private final float[][] data = new float[IJK_SIZE][];
  235. private final float[][] data1 = new float[IJK_SIZE][3];
  236. private final boolean logEnabled = logger.isLoggable(Level.FINEST);
  237. private long lastLogTime = System.currentTimeMillis();
  238. private int calls;
  239. private int missChunk;
  240. private int missSheet;
  241. private int missBlock;
  242. Chunked(IColorMap parent) {
  243. this.parent = parent;
  244. }
  245. @Override
  246. public String toString() {
  247. return parent.toString();
  248. }
  249. @Override
  250. public boolean isHeightDependent() {
  251. return parent.isHeightDependent();
  252. }
  253. @Override
  254. public int getColorMultiplier() {
  255. return parent.getColorMultiplier();
  256. }
  257. @Override
  258. public int getColorMultiplier(IBlockAccess blockAccess, int i, int j, int k) {
  259. return Colorizer.float3ToInt(getColorMultiplierF(blockAccess, i, j, k));
  260. }
  261. @Override
  262. public float[] getColorMultiplierF(IBlockAccess blockAccess, int i, int j, int k) {
  263. int offset = getChunkOffset(i, j, k);
  264. calls++;
  265. if (offset < 0) {
  266. setChunkBase(i, j, k);
  267. offset = getChunkOffset(i, j, k);
  268. missChunk++;
  269. }
  270. float[] color = data[offset];
  271. if (color == null) {
  272. color = data[offset] = data1[offset];
  273. copy3f(color, parent.getColorMultiplierF(blockAccess, i, j, k));
  274. missBlock++;
  275. }
  276. if (logEnabled) {
  277. long now = System.currentTimeMillis();
  278. if (now - lastLogTime > 5000L) {
  279. float mult = 100.0f / (float) calls;
  280. logger.finest("%s: calls: %d, miss chunk: %.2f%%, miss sheet: %.2f%%, miss block: %.2f%%",
  281. this, calls, (float) missChunk * mult, (float) missSheet * mult, (float) missBlock * mult
  282. );
  283. lastLogTime = now;
  284. }
  285. }
  286. return color;
  287. }
  288. @Override
  289. public void claimResources(Collection<ResourceLocation> resources) {
  290. parent.claimResources(resources);
  291. }
  292. @Override
  293. public IColorMap copy() {
  294. return new Chunked(parent.copy());
  295. }
  296. private static void copy3f(float[] dst, float[] src) {
  297. dst[0] = src[0];
  298. dst[1] = src[1];
  299. dst[2] = src[2];
  300. }
  301. private void setChunkBase(int i, int j, int k) {
  302. baseI = i & I_MASK;
  303. baseJ = j;
  304. baseK = k & K_MASK;
  305. baseOffset = 0;
  306. Arrays.fill(data, null);
  307. }
  308. private int getChunkOffset(int i, int j, int k) {
  309. i -= baseI;
  310. j -= baseJ;
  311. k -= baseK;
  312. if (j >= 0 && j <= J_SIZE && k >= 0 && k < K_SIZE && i >= 0 && i < I_SIZE) {
  313. if (j == J_SIZE) {
  314. j--;
  315. baseJ++;
  316. missSheet++;
  317. Arrays.fill(data, baseOffset, IJK_SIZE, null);
  318. baseOffset ^= IK_SIZE;
  319. }
  320. return (baseOffset + j * IK_SIZE + k * I_SIZE + i) % IJK_SIZE;
  321. } else {
  322. return -1;
  323. }
  324. }
  325. }
  326. static final class Outer implements IColorMap {
  327. private final IColorMap parent;
  328. private final boolean isHeightDependent;
  329. private final int mapDefault;
  330. Outer(IColorMap parent) {
  331. this.parent = parent;
  332. isHeightDependent = parent.isHeightDependent();
  333. mapDefault = parent.getColorMultiplier();
  334. }
  335. @Override
  336. public String toString() {
  337. return parent.toString();
  338. }
  339. @Override
  340. public boolean isHeightDependent() {
  341. return parent.isHeightDependent();
  342. }
  343. @Override
  344. public int getColorMultiplier() {
  345. return mapDefault;
  346. }
  347. @Override
  348. public int getColorMultiplier(IBlockAccess blockAccess, int i, int j, int k) {
  349. if (!isHeightDependent) {
  350. j = DEFAULT_HEIGHT;
  351. }
  352. return parent.getColorMultiplier(blockAccess, i, j, k);
  353. }
  354. @Override
  355. public float[] getColorMultiplierF(IBlockAccess blockAccess, int i, int j, int k) {
  356. if (!isHeightDependent) {
  357. j = DEFAULT_HEIGHT;
  358. }
  359. return parent.getColorMultiplierF(blockAccess, i, j, k);
  360. }
  361. @Override
  362. public void claimResources(Collection<ResourceLocation> resources) {
  363. parent.claimResources(resources);
  364. }
  365. @Override
  366. public IColorMap copy() {
  367. return new Outer(parent.copy());
  368. }
  369. }
  370. }