FluidEntryStack.java 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195
  1. /*
  2. * Roughly Enough Items by Danielshe.
  3. * Licensed under the MIT License.
  4. */
  5. package me.shedaniel.rei.impl;
  6. import com.google.common.collect.Lists;
  7. import me.shedaniel.math.api.Rectangle;
  8. import me.shedaniel.rei.api.ClientHelper;
  9. import me.shedaniel.rei.api.ConfigObject;
  10. import me.shedaniel.rei.api.EntryStack;
  11. import me.shedaniel.rei.gui.widget.QueuedTooltip;
  12. import net.fabricmc.fabric.api.client.render.fluid.v1.FluidRenderHandler;
  13. import net.fabricmc.fabric.api.client.render.fluid.v1.FluidRenderHandlerRegistry;
  14. import net.minecraft.client.MinecraftClient;
  15. import net.minecraft.client.render.BufferBuilder;
  16. import net.minecraft.client.render.Tessellator;
  17. import net.minecraft.client.render.VertexFormats;
  18. import net.minecraft.client.texture.Sprite;
  19. import net.minecraft.client.texture.SpriteAtlasTexture;
  20. import net.minecraft.fluid.Fluid;
  21. import net.minecraft.fluid.Fluids;
  22. import net.minecraft.util.Identifier;
  23. import net.minecraft.util.Pair;
  24. import net.minecraft.util.math.BlockPos;
  25. import net.minecraft.util.registry.Registry;
  26. import org.jetbrains.annotations.ApiStatus;
  27. import org.jetbrains.annotations.Nullable;
  28. import java.util.*;
  29. @ApiStatus.Internal
  30. public class FluidEntryStack extends AbstractEntryStack {
  31. private static final Map<Fluid, Pair<Sprite, Integer>> FLUID_SPRITE_CACHE = new HashMap<>();
  32. private static final int EMPTY_AMOUNT = -1319182373;
  33. private Fluid fluid;
  34. private int amount;
  35. public FluidEntryStack(Fluid fluid) {
  36. this(fluid, EMPTY_AMOUNT);
  37. }
  38. public FluidEntryStack(Fluid fluid, int amount) {
  39. this.fluid = fluid;
  40. this.amount = amount;
  41. }
  42. protected static Pair<Sprite, Integer> getOrLoadSprite(Fluid fluid) {
  43. Pair<Sprite, Integer> possibleCached = FLUID_SPRITE_CACHE.get(fluid);
  44. if (possibleCached != null)
  45. return possibleCached;
  46. FluidRenderHandler fluidRenderHandler = FluidRenderHandlerRegistry.INSTANCE.get(fluid);
  47. if (fluidRenderHandler == null)
  48. return null;
  49. Sprite[] sprites = fluidRenderHandler.getFluidSprites(MinecraftClient.getInstance().world, MinecraftClient.getInstance().world == null ? null : BlockPos.ORIGIN, fluid.getDefaultState());
  50. int color = -1;
  51. if (MinecraftClient.getInstance().world != null)
  52. color = fluidRenderHandler.getFluidColor(MinecraftClient.getInstance().world, BlockPos.ORIGIN, fluid.getDefaultState());
  53. Pair<Sprite, Integer> pair = new Pair<>(sprites[0], color);
  54. FLUID_SPRITE_CACHE.put(fluid, pair);
  55. return pair;
  56. }
  57. @Override
  58. public Optional<Identifier> getIdentifier() {
  59. return Optional.ofNullable(Registry.FLUID.getId(getFluid()));
  60. }
  61. @Override
  62. public Type getType() {
  63. return Type.FLUID;
  64. }
  65. @Override
  66. public int getAmount() {
  67. return amount;
  68. }
  69. @Override
  70. public void setAmount(int amount) {
  71. this.amount = amount == EMPTY_AMOUNT ? EMPTY_AMOUNT : Math.max(amount, 0);
  72. if (isEmpty()) {
  73. fluid = Fluids.EMPTY;
  74. }
  75. }
  76. @Override
  77. public boolean isEmpty() {
  78. return (amount != EMPTY_AMOUNT && amount <= 0) || fluid == Fluids.EMPTY;
  79. }
  80. @Override
  81. public EntryStack copy() {
  82. EntryStack stack = EntryStack.create(fluid, amount);
  83. for (Map.Entry<Settings<?>, Object> entry : getSettings().entrySet()) {
  84. stack.setting((Settings<? super Object>) entry.getKey(), entry.getValue());
  85. }
  86. return stack;
  87. }
  88. @Override
  89. public Object getObject() {
  90. return fluid;
  91. }
  92. @Override
  93. public boolean equalsIgnoreTagsAndAmount(EntryStack stack) {
  94. if (stack.getType() != Type.FLUID)
  95. return false;
  96. return fluid == stack.getFluid();
  97. }
  98. @Override
  99. public boolean equalsIgnoreTags(EntryStack stack) {
  100. if (stack.getType() != Type.FLUID)
  101. return false;
  102. return fluid == stack.getFluid() && amount == stack.getAmount();
  103. }
  104. @Override
  105. public boolean equalsIgnoreAmount(EntryStack stack) {
  106. if (stack.getType() != Type.FLUID)
  107. return false;
  108. return fluid == stack.getFluid();
  109. }
  110. @Override
  111. public boolean equalsAll(EntryStack stack) {
  112. if (stack.getType() != Type.FLUID)
  113. return false;
  114. return fluid == stack.getFluid() && amount == stack.getAmount();
  115. }
  116. @Override
  117. public int hashCode() {
  118. int result = 1;
  119. result = 31 * result + getType().ordinal();
  120. result = 31 * result + fluid.hashCode();
  121. result = 31 * result + amount;
  122. result = 31 * result;
  123. return result;
  124. }
  125. @Nullable
  126. @Override
  127. public QueuedTooltip getTooltip(int mouseX, int mouseY) {
  128. if (!get(Settings.TOOLTIP_ENABLED).get() || isEmpty())
  129. return null;
  130. List<String> toolTip = Lists.newArrayList(SearchArgument.tryGetEntryStackName(this));
  131. if (amount >= 0) {
  132. String amountTooltip = get(Settings.Fluid.AMOUNT_TOOLTIP).apply(this);
  133. if (amountTooltip != null)
  134. toolTip.addAll(Arrays.asList(amountTooltip.split("\n")));
  135. }
  136. toolTip.addAll(get(Settings.TOOLTIP_APPEND_EXTRA).apply(this));
  137. if (get(Settings.TOOLTIP_APPEND_MOD).get() && ConfigObject.getInstance().shouldAppendModNames()) {
  138. final String modString = ClientHelper.getInstance().getFormattedModFromIdentifier(Registry.FLUID.getId(fluid));
  139. boolean alreadyHasMod = false;
  140. for (String s : toolTip)
  141. if (s.equalsIgnoreCase(modString)) {
  142. alreadyHasMod = true;
  143. break;
  144. }
  145. if (!alreadyHasMod)
  146. toolTip.add(modString);
  147. }
  148. return QueuedTooltip.create(toolTip);
  149. }
  150. @Override
  151. public void render(Rectangle bounds, int mouseX, int mouseY, float delta) {
  152. if (get(Settings.RENDER).get()) {
  153. Pair<Sprite, Integer> pair = getOrLoadSprite(getFluid());
  154. if (pair != null) {
  155. Sprite sprite = pair.getLeft();
  156. int color = pair.getRight();
  157. int a = 255;
  158. int r = (color >> 16 & 255);
  159. int g = (color >> 8 & 255);
  160. int b = (color & 255);
  161. MinecraftClient.getInstance().getTextureManager().bindTexture(SpriteAtlasTexture.BLOCK_ATLAS_TEX);
  162. Tessellator tess = Tessellator.getInstance();
  163. BufferBuilder bb = tess.getBuffer();
  164. bb.begin(7, VertexFormats.POSITION_TEXTURE_COLOR);
  165. bb.vertex(bounds.getMaxX(), bounds.y, getZ()).texture(sprite.getMaxU(), sprite.getMinV()).color(r, g, b, a).next();
  166. bb.vertex(bounds.x, bounds.y, getZ()).texture(sprite.getMinU(), sprite.getMinV()).color(r, g, b, a).next();
  167. bb.vertex(bounds.x, bounds.getMaxY(), getZ()).texture(sprite.getMinU(), sprite.getMaxV()).color(r, g, b, a).next();
  168. bb.vertex(bounds.getMaxX(), bounds.getMaxY(), getZ()).texture(sprite.getMaxU(), sprite.getMaxV()).color(r, g, b, a).next();
  169. tess.draw();
  170. }
  171. }
  172. }
  173. }