minecraft-src/net/minecraft/client/renderer/item/BlockModelWrapper.java
2025-07-04 03:15:13 +03:00

89 lines
3.1 KiB
Java

package net.minecraft.client.renderer.item;
import com.mojang.serialization.MapCodec;
import com.mojang.serialization.codecs.RecordCodecBuilder;
import java.util.List;
import net.fabricmc.api.EnvType;
import net.fabricmc.api.Environment;
import net.minecraft.client.color.item.ItemTintSource;
import net.minecraft.client.color.item.ItemTintSources;
import net.minecraft.client.multiplayer.ClientLevel;
import net.minecraft.client.renderer.ItemBlockRenderTypes;
import net.minecraft.client.renderer.RenderType;
import net.minecraft.client.resources.model.BakedModel;
import net.minecraft.client.resources.model.ResolvableModel;
import net.minecraft.resources.ResourceLocation;
import net.minecraft.tags.ItemTags;
import net.minecraft.world.entity.LivingEntity;
import net.minecraft.world.item.ItemDisplayContext;
import net.minecraft.world.item.ItemStack;
import net.minecraft.world.item.Items;
import org.jetbrains.annotations.Nullable;
@Environment(EnvType.CLIENT)
public class BlockModelWrapper implements ItemModel {
private final BakedModel model;
private final List<ItemTintSource> tints;
BlockModelWrapper(BakedModel model, List<ItemTintSource> tints) {
this.model = model;
this.tints = tints;
}
@Override
public void update(
ItemStackRenderState renderState,
ItemStack stack,
ItemModelResolver itemModelResolver,
ItemDisplayContext displayContext,
@Nullable ClientLevel level,
@Nullable LivingEntity entity,
int seed
) {
ItemStackRenderState.LayerRenderState layerRenderState = renderState.newLayer();
if (stack.hasFoil()) {
layerRenderState.setFoilType(hasSpecialAnimatedTexture(stack) ? ItemStackRenderState.FoilType.SPECIAL : ItemStackRenderState.FoilType.STANDARD);
}
int i = this.tints.size();
int[] is = layerRenderState.prepareTintLayers(i);
for (int j = 0; j < i; j++) {
is[j] = ((ItemTintSource)this.tints.get(j)).calculate(stack, level, entity);
}
RenderType renderType = ItemBlockRenderTypes.getRenderType(stack);
layerRenderState.setupBlockModel(this.model, renderType);
}
private static boolean hasSpecialAnimatedTexture(ItemStack stack) {
return stack.is(ItemTags.COMPASSES) || stack.is(Items.CLOCK);
}
@Environment(EnvType.CLIENT)
public record Unbaked(ResourceLocation model, List<ItemTintSource> tints) implements ItemModel.Unbaked {
public static final MapCodec<BlockModelWrapper.Unbaked> MAP_CODEC = RecordCodecBuilder.mapCodec(
instance -> instance.group(
ResourceLocation.CODEC.fieldOf("model").forGetter(BlockModelWrapper.Unbaked::model),
ItemTintSources.CODEC.listOf().optionalFieldOf("tints", List.of()).forGetter(BlockModelWrapper.Unbaked::tints)
)
.apply(instance, BlockModelWrapper.Unbaked::new)
);
@Override
public void resolveDependencies(ResolvableModel.Resolver resolver) {
resolver.resolve(this.model);
}
@Override
public ItemModel bake(ItemModel.BakingContext context) {
BakedModel bakedModel = context.bake(this.model);
return new BlockModelWrapper(bakedModel, this.tints);
}
@Override
public MapCodec<BlockModelWrapper.Unbaked> type() {
return MAP_CODEC;
}
}
}