tutorial:colorprovider
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
tutorial:colorprovider [2021/01/22 16:49] – [Registering a Block Color Provider] ChunkRendererRegions are actually 20x20x20 comp500 | tutorial:colorprovider [2025/04/01 12:10] (current) – solidblock | ||
---|---|---|---|
Line 1: | Line 1: | ||
- | ====== Color Providers ====== | + | ======= Color Providers |
- | Ever wonder how grass and leaves change hues depending on the biome, or how leather armor can have seemingly infinite color patterns? Meet color providers, which allow you to hue and tint block & item model textures based on properties such as location, NBT, or block states. | + | Ever wonder how grass and leaves change hues depending on the biome, or how leather armor can have seemingly infinite color patterns? Meet **color providers**, which allow you to hue and tint block & item model textures based on properties such as location, NBT, or block states. |
- | === Existing | + | ===== Vanilla |
First, what existing vanilla content uses color providers? A few examples include: | First, what existing vanilla content uses color providers? A few examples include: | ||
* grass | * grass | ||
Line 8: | Line 8: | ||
* leather armor dying | * leather armor dying | ||
* redstone wire | * redstone wire | ||
- | * plants such as melons, sugarcane, and lilypads | + | * plants such as melons, sugarcane, and lily pads |
* tipped arrows | * tipped arrows | ||
Line 17: | Line 17: | ||
Remember that the color provider is a client-side mechanic. Make sure to put any code related to it inside a client initializer. | Remember that the color provider is a client-side mechanic. Make sure to put any code related to it inside a client initializer. | ||
- | ===== Registering a Block Color Provider ===== | + | ===== Block Color Provider ===== |
- | To register a block to the block color provider, you'll need to use Fabric' | + | To register a block to the block color provider, you'll need to use Fabric' |
- | <code java [enable_line_numbers="false"]> | + | |
- | ColorProviderRegistry.BLOCK.register((state, view, pos, tintIndex) -> 0x3495eb, MY_BLOCK); | + | At first, we create the block in '' |
+ | <code java TutorialBlocks.class> | ||
+ | public final class TutorialBlocks { | ||
+ | | ||
+ | // Before 1.21.2 | ||
+ | public static final Block COLOR_BLOCK | ||
+ | |||
+ | // 1.21.2 and after: | ||
+ | public static final Block COLOR_BLOCK = register(" | ||
+ | } | ||
</ | </ | ||
- | All we do here is say, "Hi, '' | + | Then add a simple block states file: |
+ | <code javascript src/ | ||
+ | { | ||
+ | " | ||
+ | "": | ||
+ | " | ||
+ | } | ||
+ | } | ||
+ | } | ||
+ | </ | ||
- | If you need to access | + | In your '' |
+ | <code java ExampleModClient.java> | ||
+ | @Environment(EnvType.CLIENT) | ||
+ | public class ExampleModClient implements ClientModInitializer { | ||
+ | @Override | ||
+ | public void onInitializeClient() { | ||
+ | | ||
- | The model is also important: the main note here is that you are // | + | ColorProviderRegistry.BLOCK.register((state, |
- | <code json [enable_line_numbers=" | + | } |
+ | } | ||
+ | </ | ||
+ | |||
+ | If you haven' | ||
+ | <code javascript> | ||
+ | { | ||
+ | // ... | ||
+ | " | ||
+ | // ... | ||
+ | " | ||
+ | " | ||
+ | ] | ||
+ | }, | ||
+ | // ... | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | Then we create the block model with // | ||
+ | < | ||
{ | { | ||
" | " | ||
Line 36: | Line 79: | ||
}, | }, | ||
" | " | ||
- | { | + | { " |
" | " | ||
" | " | ||
Line 50: | Line 93: | ||
} | } | ||
</ | </ | ||
- | In this instance, we're adding a single tintindex, which is what would appear in the '' | + | |
+ | In this instance, we're adding a single tintindex, which is what would appear in the '' | ||
+ | <code javascript src/ | ||
+ | { | ||
+ | " | ||
+ | " | ||
+ | " | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | > **Note:** the color of the block is cached. If you privded a color that changes among time, the changes will not take effect immediately unless there are block updates nearby. | ||
Here's the final result-- note that the original model used the '' | Here's the final result-- note that the original model used the '' | ||
{{https:// | {{https:// | ||
- | ===== Registering an Item Color Provider ===== | + | ===== Block Entity with Color Provider ===== |
- | Items are similar; the difference | + | |
- | < | + | If you need to access '' |
- | ColorProviderRegistry.ITEM.register((stack, | + | |
+ | This is because blocks can be rendered on separate threads, so accessing the data directly is not safe. Additionally, | ||
+ | |||
+ | In this case, we create a '' | ||
+ | |||
+ | <code java ColorBlock.java> | ||
+ | public class ColorBlock extends BlockWithEntity { | ||
+ | public ColorBlock(Settings settings) { | ||
+ | super(settings); | ||
+ | } | ||
+ | |||
+ | @Override | ||
+ | protected MapCodec<? | ||
+ | return createCodec(ColorBlock:: | ||
+ | } | ||
+ | |||
+ | @Nullable | ||
+ | @Override | ||
+ | public BlockEntity createBlockEntity(BlockPos pos, BlockState state) { | ||
+ | return new ColorBlockEntity(pos, | ||
+ | } | ||
+ | |||
+ | // Since 1.21.4, this method is not required anymore, because all block entities use their block model by default. | ||
+ | @Override | ||
+ | protected BlockRenderType getRenderType(BlockState state) { | ||
+ | return BlockRenderType.MODEL; | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | <code java ColorBlockEntity.java> | ||
+ | public class ColorBlockEntity extends BlockEntity { | ||
+ | public int color = 0x3495eb; | ||
+ | |||
+ | public ColorBlockEntity(BlockPos pos, BlockState state) { | ||
+ | super(TutorialBlockEntityTypes.COLOR_BLOCK, | ||
+ | } | ||
+ | |||
+ | // The following two methods specify serialization of color data. | ||
+ | |||
+ | @Override | ||
+ | protected void readNbt(NbtCompound nbt, RegistryWrapper.WrapperLookup registryLookup) { | ||
+ | super.readNbt(nbt, | ||
+ | |||
+ | // For versions before 1.21.5, please directly use nbt.getInt(" | ||
+ | color = nbt.getInt(" | ||
+ | |||
+ | // When the data is modified through "/ | ||
+ | // or placed by an item with " | ||
+ | // the render color will be updated. | ||
+ | if (world != null) { | ||
+ | world.updateListeners(pos, | ||
+ | } | ||
+ | } | ||
+ | |||
+ | @Override | ||
+ | protected void writeNbt(NbtCompound nbt, RegistryWrapper.WrapperLookup registryLookup) { | ||
+ | super.writeNbt(nbt, | ||
+ | nbt.putInt(" | ||
+ | } | ||
+ | |||
+ | @Nullable | ||
+ | @Override | ||
+ | public Packet< | ||
+ | return BlockEntityUpdateS2CPacket.create(this); | ||
+ | } | ||
+ | |||
+ | @Override | ||
+ | public NbtCompound toInitialChunkDataNbt(RegistryWrapper.WrapperLookup registryLookup) { | ||
+ | return createNbt(registryLookup); | ||
+ | } | ||
+ | |||
+ | @Override | ||
+ | public @Nullable Object getRenderData() { | ||
+ | // this is the method from `RenderDataBlockEntity` class. | ||
+ | return color; | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | In the '' | ||
+ | <code java> | ||
+ | // Before 1.21.2: | ||
+ | public static final ColorBlock COLOR_BLOCK = register(" | ||
+ | |||
+ | // Since 1.21.2: | ||
+ | public static final Block COLOR_BLOCK = register(" | ||
+ | </ | ||
+ | |||
+ | In the '' | ||
+ | <code java> | ||
+ | public static final BlockEntityType< | ||
+ | BlockEntityType.Builder.create(ColorBlockEntity:: | ||
+ | </ | ||
+ | |||
+ | Now we modify '' | ||
+ | <code java ColorBlock.java> | ||
+ | public class ColorBlock extends BlockWithEntity { | ||
+ | [...] | ||
+ | |||
+ | @Override | ||
+ | protected ActionResult onUseWithItem(ItemStack stack, BlockState state, World world, BlockPos pos, PlayerEntity player, Hand hand, BlockHitResult hit) { | ||
+ | if (stack.getItem() instanceof DyeItem dyeItem) { | ||
+ | if (world.getBlockEntity(pos) instanceof ColorBlockEntity colorBlockEntity) { | ||
+ | final int newColor = dyeItem.getColor().getEntityColor(); | ||
+ | final int originalColor = colorBlockEntity.color; | ||
+ | colorBlockEntity.color = ColorHelper.average(newColor, | ||
+ | stack.decrementUnlessCreative(1, | ||
+ | colorBlockEntity.markDirty(); | ||
+ | world.updateListeners(pos, | ||
+ | } | ||
+ | } | ||
+ | return super.onUseWithItem(stack, | ||
+ | } | ||
+ | |||
+ | [...] | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | Finally, modify the color provider to use the render data. We call '' | ||
+ | <code java ExampleModClient.java> | ||
+ | @Environment(EnvType.CLIENT) | ||
+ | public class ExampleModClient implements ClientModInitializer { | ||
+ | [...] | ||
+ | @Override | ||
+ | public void onInitializeClient() { | ||
+ | [...] | ||
+ | |||
+ | ColorProviderRegistry.BLOCK.register((state, | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | Now done! Then you can check whether the following work correctly: | ||
+ | * When you interact the block with a dye, the color should change. | ||
+ | * When you modify the color through ''/ | ||
+ | * When you pick (press mouse wheel) the block with '' | ||
+ | * When you leave the world and re-enter, the color should be kept. | ||
+ | |||
+ | ===== Custom item tint (1.21.4 and after) ===== | ||
+ | Since 1.21.4 开始, tints of items are defined in item models definitions. Some common tint source types are provided | ||
+ | <code javascript / | ||
+ | { | ||
+ | " | ||
+ | " | ||
+ | " | ||
+ | " | ||
+ | { | ||
+ | " | ||
+ | " | ||
+ | } | ||
+ | ] | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | If you need to specify a custom tint source, you can register with '' | ||
+ | |||
+ | > If the tint does not work, check the value of tintindex in the model, which should be consistent with the element subscript in the '' | ||
+ | |||
+ | In version 1.21.3 and before, item model providers are also registered with Fabric API. Different from blocks, the context provided, instead | ||
+ | |||
+ | For item models, we can directly inherite the block model that uses tintindex: | ||
+ | < | ||
+ | { | ||
+ | | ||
+ | } | ||
+ | </code> | ||
+ | |||
+ | <code java ExampleModClient.java> | ||
+ | @Environment(EnvType.CLIENT) | ||
+ | public class ExampleModClient implements ClientModInitializer { | ||
+ | @Override | ||
+ | public void onInitializeClient() { | ||
+ | // ... | ||
+ | |||
+ | | ||
+ | } | ||
+ | } | ||
</ | </ | ||
This would hue the item in our inventory in the same fashion as the block. | This would hue the item in our inventory in the same fashion as the block. | ||
- | === Limitations === | + | ===== Limitations |
One key issue with using the color provider is the lack of context in the item provider. This is why vanilla grass doesn' | One key issue with using the color provider is the lack of context in the item provider. This is why vanilla grass doesn' |
tutorial/colorprovider.1611334167.txt.gz · Last modified: 2021/01/22 16:49 by comp500