/* * This file is licensed under the MIT License, part of Roughly Enough Items. * Copyright (c) 2018, 2019, 2020 shedaniel * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all * copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ package me.shedaniel.rei.impl; import com.google.common.collect.Lists; import me.shedaniel.math.Point; import me.shedaniel.math.Rectangle; import me.shedaniel.rei.api.ClientHelper; import me.shedaniel.rei.api.ConfigObject; import me.shedaniel.rei.api.EntryStack; import me.shedaniel.rei.api.widgets.Tooltip; import net.fabricmc.fabric.api.client.render.fluid.v1.FluidRenderHandler; import net.fabricmc.fabric.api.client.render.fluid.v1.FluidRenderHandlerRegistry; import net.minecraft.client.MinecraftClient; import net.minecraft.client.render.BufferBuilder; import net.minecraft.client.render.Tessellator; import net.minecraft.client.render.VertexFormats; import net.minecraft.client.texture.Sprite; import net.minecraft.client.texture.SpriteAtlasTexture; import net.minecraft.fluid.Fluid; import net.minecraft.fluid.Fluids; import net.minecraft.util.Identifier; import net.minecraft.util.Pair; import net.minecraft.util.math.BlockPos; import net.minecraft.util.registry.Registry; import org.jetbrains.annotations.ApiStatus; import org.jetbrains.annotations.Nullable; import java.util.*; @ApiStatus.Internal public class FluidEntryStack extends AbstractEntryStack { private static final Map> FLUID_SPRITE_CACHE = new HashMap<>(); private static final int EMPTY_AMOUNT = -1319182373; private Fluid fluid; private int amount; public FluidEntryStack(Fluid fluid) { this(fluid, EMPTY_AMOUNT); } public FluidEntryStack(Fluid fluid, int amount) { this.fluid = fluid; this.amount = amount; } protected static Pair getOrLoadSprite(Fluid fluid) { Pair possibleCached = FLUID_SPRITE_CACHE.get(fluid); if (possibleCached != null) return possibleCached; FluidRenderHandler fluidRenderHandler = FluidRenderHandlerRegistry.INSTANCE.get(fluid); if (fluidRenderHandler == null) return null; Sprite[] sprites = fluidRenderHandler.getFluidSprites(MinecraftClient.getInstance().world, MinecraftClient.getInstance().world == null ? null : BlockPos.ORIGIN, fluid.getDefaultState()); int color = -1; if (MinecraftClient.getInstance().world != null) color = fluidRenderHandler.getFluidColor(MinecraftClient.getInstance().world, BlockPos.ORIGIN, fluid.getDefaultState()); Pair pair = new Pair<>(sprites[0], color); FLUID_SPRITE_CACHE.put(fluid, pair); return pair; } @Override public Optional getIdentifier() { return Optional.ofNullable(Registry.FLUID.getId(getFluid())); } @Override public Type getType() { return Type.FLUID; } @Override public int getAmount() { return amount; } @Override public void setAmount(int amount) { this.amount = amount == EMPTY_AMOUNT ? EMPTY_AMOUNT : Math.max(amount, 0); if (isEmpty()) { fluid = Fluids.EMPTY; } } @Override public boolean isEmpty() { return (amount != EMPTY_AMOUNT && amount <= 0) || fluid == Fluids.EMPTY; } @Override public EntryStack copy() { EntryStack stack = EntryStack.create(fluid, amount); for (Map.Entry, Object> entry : getSettings().entrySet()) { stack.setting((Settings) entry.getKey(), entry.getValue()); } return stack; } @Override public Object getObject() { return fluid; } @Override public boolean equalsIgnoreTagsAndAmount(EntryStack stack) { if (stack.getType() != Type.FLUID) return false; return fluid == stack.getFluid(); } @Override public boolean equalsIgnoreTags(EntryStack stack) { if (stack.getType() != Type.FLUID) return false; return fluid == stack.getFluid() && amount == stack.getAmount(); } @Override public boolean equalsIgnoreAmount(EntryStack stack) { if (stack.getType() != Type.FLUID) return false; return fluid == stack.getFluid(); } @Override public boolean equalsAll(EntryStack stack) { if (stack.getType() != Type.FLUID) return false; return fluid == stack.getFluid() && amount == stack.getAmount(); } @Override public int hashOfAll() { int result = hashIgnoreAmountAndTags(); result = 31 * result + amount; return result; } @Override public int hashIgnoreAmountAndTags() { int result = 1; result = 31 * result + getType().ordinal(); result = 31 * result + fluid.hashCode(); return result; } @Override public int hashIgnoreTags() { return hashOfAll(); } @Override public int hashIgnoreAmount() { return hashIgnoreAmountAndTags(); } @Nullable @Override public Tooltip getTooltip(Point point) { if (!get(Settings.TOOLTIP_ENABLED).get() || isEmpty()) return null; List toolTip = Lists.newArrayList(SearchArgument.tryGetEntryStackName(this)); if (amount >= 0) { String amountTooltip = get(Settings.Fluid.AMOUNT_TOOLTIP).apply(this); if (amountTooltip != null) toolTip.addAll(Arrays.asList(amountTooltip.split("\n"))); } toolTip.addAll(get(Settings.TOOLTIP_APPEND_EXTRA).apply(this)); if (get(Settings.TOOLTIP_APPEND_MOD).get() && ConfigObject.getInstance().shouldAppendModNames()) { final String modString = ClientHelper.getInstance().getFormattedModFromIdentifier(Registry.FLUID.getId(fluid)); boolean alreadyHasMod = false; for (String s : toolTip) if (s.equalsIgnoreCase(modString)) { alreadyHasMod = true; break; } if (!alreadyHasMod) toolTip.add(modString); } return Tooltip.create(toolTip); } @SuppressWarnings("deprecation") @Override public void render(Rectangle bounds, int mouseX, int mouseY, float delta) { if (get(Settings.RENDER).get()) { Pair pair = getOrLoadSprite(getFluid()); if (pair != null) { Sprite sprite = pair.getLeft(); int color = pair.getRight(); int a = 255; int r = (color >> 16 & 255); int g = (color >> 8 & 255); int b = (color & 255); MinecraftClient.getInstance().getTextureManager().bindTexture(SpriteAtlasTexture.BLOCK_ATLAS_TEX); Tessellator tess = Tessellator.getInstance(); BufferBuilder bb = tess.getBuffer(); bb.begin(7, VertexFormats.POSITION_TEXTURE_COLOR); bb.vertex(bounds.getMaxX(), bounds.y, getZ()).texture(sprite.getMaxU(), sprite.getMinV()).color(r, g, b, a).next(); bb.vertex(bounds.x, bounds.y, getZ()).texture(sprite.getMinU(), sprite.getMinV()).color(r, g, b, a).next(); bb.vertex(bounds.x, bounds.getMaxY(), getZ()).texture(sprite.getMinU(), sprite.getMaxV()).color(r, g, b, a).next(); bb.vertex(bounds.getMaxX(), bounds.getMaxY(), getZ()).texture(sprite.getMaxU(), sprite.getMaxV()).color(r, g, b, a).next(); tess.draw(); } } } }