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

https://bitbucket.org/SevenBits/mcpatcher · Java · 139 lines · 126 code · 13 blank · 0 comment · 34 complexity · 770c9dd924a02af8836f6fc31b27f4ec MD5 · raw file

  1. package com.prupe.mcpatcher.cc;
  2. import com.prupe.mcpatcher.Config;
  3. import com.prupe.mcpatcher.MCLogger;
  4. import com.prupe.mcpatcher.MCPatcherUtils;
  5. import com.prupe.mcpatcher.TexturePackAPI;
  6. import net.minecraft.client.Minecraft;
  7. import net.minecraft.src.EntityRenderer;
  8. import net.minecraft.src.ResourceLocation;
  9. import net.minecraft.src.World;
  10. import java.awt.image.BufferedImage;
  11. import java.util.HashMap;
  12. public final class Lightmap {
  13. private static final MCLogger logger = MCLogger.getLogger(MCPatcherUtils.CUSTOM_COLORS);
  14. private static final String LIGHTMAP_FORMAT = "lightmap/world%d.png";
  15. private static final int LIGHTMAP_SIZE = 16;
  16. private static final int HEIGHT_WITHOUT_NIGHTVISION = 2 * LIGHTMAP_SIZE;
  17. private static final int HEIGHT_WITH_NIGHTVISION = 4 * LIGHTMAP_SIZE;
  18. private static final boolean useLightmaps = Config.getBoolean(MCPatcherUtils.CUSTOM_COLORS, "lightmaps", true);
  19. private static final HashMap<Integer, Lightmap> lightmaps = new HashMap<Integer, Lightmap>();
  20. private final int width;
  21. private final boolean customNightvision;
  22. private final int[] origMap;
  23. private final boolean valid;
  24. private final float[] sunrgb = new float[3 * LIGHTMAP_SIZE];
  25. private final float[] torchrgb = new float[3 * LIGHTMAP_SIZE];
  26. private final float[] sunrgbnv = new float[3 * LIGHTMAP_SIZE];
  27. private final float[] torchrgbnv = new float[3 * LIGHTMAP_SIZE];
  28. private final float[] rgb = new float[3];
  29. static void reset() {
  30. lightmaps.clear();
  31. }
  32. public static boolean computeLightmap(EntityRenderer renderer, World world, int[] mapRGB, float partialTick) {
  33. if (world == null || !useLightmaps) {
  34. return false;
  35. }
  36. Lightmap lightmap = null;
  37. int worldType = world.worldProvider.getWorldType();
  38. if (lightmaps.containsKey(worldType)) {
  39. lightmap = lightmaps.get(worldType);
  40. } else {
  41. ResourceLocation resource = TexturePackAPI.newMCPatcherResourceLocation(String.format(LIGHTMAP_FORMAT, worldType));
  42. BufferedImage image = TexturePackAPI.getImage(resource);
  43. if (image != null) {
  44. lightmap = new Lightmap(resource, image);
  45. if (!lightmap.valid) {
  46. lightmap = null;
  47. }
  48. }
  49. lightmaps.put(worldType, lightmap);
  50. }
  51. return lightmap != null && lightmap.compute(renderer, world, mapRGB, partialTick);
  52. }
  53. private Lightmap(ResourceLocation resource, BufferedImage image) {
  54. width = image.getWidth();
  55. int height = image.getHeight();
  56. customNightvision = (height == HEIGHT_WITH_NIGHTVISION);
  57. origMap = new int[width * height];
  58. image.getRGB(0, 0, width, height, origMap, 0, width);
  59. valid = (height == HEIGHT_WITHOUT_NIGHTVISION || height == HEIGHT_WITH_NIGHTVISION);
  60. if (!valid) {
  61. logger.error("%s must be exactly %d or %d pixels high", resource, HEIGHT_WITHOUT_NIGHTVISION, HEIGHT_WITH_NIGHTVISION);
  62. }
  63. }
  64. private boolean compute(EntityRenderer renderer, World world, int[] mapRGB, float partialTick) {
  65. float sun = Colorizer.clamp(world.lightningFlash > 0 ? 1.0f : 7.0f / 6.0f * (world.getSunAngle(1.0f) - 0.2f)) * (width - 1);
  66. float torch = Colorizer.clamp(renderer.torchFlickerX + 0.5f) * (width - 1);
  67. float nightVisionStrength = renderer.getNightVisionStrength(partialTick);
  68. float gamma = Colorizer.clamp(Minecraft.getInstance().gameSettings.gammaSetting);
  69. for (int i = 0; i < LIGHTMAP_SIZE; i++) {
  70. interpolate(origMap, i * width, sun, sunrgb, 3 * i);
  71. interpolate(origMap, (i + LIGHTMAP_SIZE) * width, torch, torchrgb, 3 * i);
  72. if (customNightvision && nightVisionStrength > 0.0f) {
  73. interpolate(origMap, (i + 2 * LIGHTMAP_SIZE) * width, sun, sunrgbnv, 3 * i);
  74. interpolate(origMap, (i + 3 * LIGHTMAP_SIZE) * width, torch, torchrgbnv, 3 * i);
  75. }
  76. }
  77. for (int s = 0; s < LIGHTMAP_SIZE; s++) {
  78. for (int t = 0; t < LIGHTMAP_SIZE; t++) {
  79. for (int k = 0; k < 3; k++) {
  80. rgb[k] = Colorizer.clamp(sunrgb[3 * s + k] + torchrgb[3 * t + k]);
  81. }
  82. if (nightVisionStrength > 0.0f) {
  83. if (customNightvision) {
  84. for (int k = 0; k < 3; k++) {
  85. rgb[k] = Colorizer.clamp((1.0f - nightVisionStrength) * rgb[k] + nightVisionStrength * (sunrgbnv[3 * s + k] + torchrgbnv[3 * t + k]));
  86. }
  87. } else {
  88. float nightVisionMultiplier = Math.max(Math.max(rgb[0], rgb[1]), rgb[2]);
  89. if (nightVisionMultiplier > 0.0f) {
  90. nightVisionMultiplier = (1.0f - nightVisionStrength) + nightVisionStrength / nightVisionMultiplier;
  91. for (int k = 0; k < 3; k++) {
  92. rgb[k] = Colorizer.clamp(rgb[k] * nightVisionMultiplier);
  93. }
  94. }
  95. }
  96. }
  97. if (gamma != 0.0f) {
  98. for (int k = 0; k < 3; k++) {
  99. float tmp = 1.0f - rgb[k];
  100. tmp = 1.0f - tmp * tmp * tmp * tmp;
  101. rgb[k] = gamma * tmp + (1.0f - gamma) * rgb[k];
  102. }
  103. }
  104. mapRGB[s * LIGHTMAP_SIZE + t] = 0xff000000 | Colorizer.float3ToInt(rgb);
  105. }
  106. }
  107. return true;
  108. }
  109. private static void interpolate(int[] map, int offset1, float x, float[] rgb, int offset2) {
  110. int x0 = (int) Math.floor(x);
  111. int x1 = (int) Math.ceil(x);
  112. if (x0 == x1) {
  113. Colorizer.intToFloat3(map[offset1 + x0], rgb, offset2);
  114. } else {
  115. float xf = x - x0;
  116. float xg = 1.0f - xf;
  117. float[] rgb0 = new float[3];
  118. float[] rgb1 = new float[3];
  119. Colorizer.intToFloat3(map[offset1 + x0], rgb0);
  120. Colorizer.intToFloat3(map[offset1 + x1], rgb1);
  121. for (int i = 0; i < 3; i++) {
  122. rgb[offset2 + i] = xg * rgb0[i] + xf * rgb1[i];
  123. }
  124. }
  125. }
  126. }