mirror of
https://github.com/FabricMC/fabric.git
synced 2025-04-03 10:39:57 -04:00
Port to 24w20a (#3772)
* Port to 24w20a * Update yarn * Fix TODOs * Bump version --------- Co-authored-by: Drex <nicknamedrex@gmail.com>
This commit is contained in:
parent
5e47b9cbbd
commit
e633f8839b
27 changed files with 110 additions and 224 deletions
fabric-data-attachment-api-v1/src/testmod/java/net/fabricmc/fabric/test/attachment/gametest
fabric-dimensions-v1/src
main
java/net/fabricmc/fabric
api/dimension/v1
impl/dimension
mixin/dimension
resources
testmod/java/net/fabricmc/fabric/test/dimension
fabric-entity-events-v1/src/main/java/net/fabricmc/fabric/mixin/entity/event
fabric-events-interaction-v0/src/client/java/net/fabricmc/fabric/mixin/event/interaction/client
fabric-item-api-v1/src/main/java/net/fabricmc/fabric/mixin/item
fabric-lifecycle-events-v1/src/main
java/net/fabricmc/fabric/mixin/event/lifecycle
resources
fabric-networking-api-v1/src/main
java/net/fabricmc/fabric
api/networking/v1
mixin/networking/accessor
resources
fabric-object-builder-api-v1/src/main/java/net/fabricmc/fabric
api/object/builder/v1/block
mixin/object/builder
fabric-rendering-v1/src/client/java/net/fabricmc/fabric
api/client/rendering/v1
impl/client/rendering
mixin/client/rendering
|
@ -28,6 +28,7 @@ import net.minecraft.test.GameTest;
|
|||
import net.minecraft.test.GameTestException;
|
||||
import net.minecraft.test.TestContext;
|
||||
import net.minecraft.util.Identifier;
|
||||
import net.minecraft.world.TeleportTarget;
|
||||
import net.minecraft.world.World;
|
||||
|
||||
import net.fabricmc.fabric.api.attachment.v1.AttachmentRegistry;
|
||||
|
@ -56,7 +57,7 @@ public class AttachmentCopyTests implements FabricGameTest {
|
|||
entity.setAttached(DUMMY, () -> 10);
|
||||
entity.setAttached(COPY_ON_DEATH, () -> 10);
|
||||
|
||||
Entity moved = entity.moveToWorld(end);
|
||||
Entity moved = entity.moveToWorld(() -> new TeleportTarget(end));
|
||||
if (moved == null) throw new GameTestException("Cross-world teleportation failed");
|
||||
|
||||
IntSupplier attached1 = moved.getAttached(DUMMY);
|
||||
|
|
|
@ -20,7 +20,6 @@ import com.google.common.base.Preconditions;
|
|||
import org.jetbrains.annotations.Nullable;
|
||||
|
||||
import net.minecraft.entity.Entity;
|
||||
import net.minecraft.server.world.ServerWorld;
|
||||
import net.minecraft.world.TeleportTarget;
|
||||
|
||||
import net.fabricmc.fabric.impl.dimension.FabricDimensionInternals;
|
||||
|
@ -36,16 +35,12 @@ public final class FabricDimensions {
|
|||
/**
|
||||
* Teleports an entity to a different dimension, placing it at the specified destination.
|
||||
*
|
||||
* <p>Using this method will circumvent Vanilla's portal placement code.
|
||||
*
|
||||
* <p>When teleporting to another dimension, the entity may be replaced with a new entity in the target
|
||||
* dimension. This is not the case for players, but needs to be accounted for by the caller.
|
||||
*
|
||||
* @param teleported the entity to teleport
|
||||
* @param destination the dimension the entity will be teleported to
|
||||
* @param target where the entity will be placed in the target world.
|
||||
* As in Vanilla, the target's velocity is not applied to players.
|
||||
* If target is null, the entity will not be teleported.
|
||||
* @param <E> the type of the teleported entity
|
||||
* @return Returns the teleported entity in the target dimension, which may be a new entity or <code>teleported</code>,
|
||||
* depending on the entity type.
|
||||
|
@ -53,10 +48,10 @@ public final class FabricDimensions {
|
|||
* @apiNote this method must be called from the main server thread
|
||||
*/
|
||||
@Nullable
|
||||
public static <E extends Entity> E teleport(E teleported, ServerWorld destination, TeleportTarget target) {
|
||||
public static <E extends Entity> E teleport(E teleported, TeleportTarget target) {
|
||||
Preconditions.checkNotNull(target, "A target must be provided");
|
||||
Preconditions.checkState(!teleported.getWorld().isClient, "Entities can only be teleported on the server side");
|
||||
|
||||
return FabricDimensionInternals.changeDimension(teleported, destination, target);
|
||||
return FabricDimensionInternals.changeDimension(teleported, target);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -29,30 +29,24 @@ public final class FabricDimensionInternals {
|
|||
}
|
||||
|
||||
@SuppressWarnings("unchecked")
|
||||
public static <E extends Entity> E changeDimension(E teleported, ServerWorld dimension, TeleportTarget target) {
|
||||
public static <E extends Entity> E changeDimension(E teleported, TeleportTarget target) {
|
||||
Preconditions.checkArgument(!teleported.getWorld().isClient, "Entities can only be teleported on the server side");
|
||||
Preconditions.checkArgument(Thread.currentThread() == ((ServerWorld) teleported.getWorld()).getServer().getThread(), "Entities must be teleported from the main server thread");
|
||||
|
||||
try {
|
||||
((Teleportable) teleported).fabric_setCustomTeleportTarget(target);
|
||||
|
||||
// Fast path for teleporting within the same dimension.
|
||||
if (teleported.getWorld() == dimension) {
|
||||
if (teleported instanceof ServerPlayerEntity serverPlayerEntity) {
|
||||
serverPlayerEntity.networkHandler.requestTeleport(target.position.x, target.position.y, target.position.z, target.yaw, target.pitch);
|
||||
} else {
|
||||
teleported.refreshPositionAndAngles(target.position.x, target.position.y, target.position.z, target.yaw, target.pitch);
|
||||
}
|
||||
|
||||
teleported.setVelocity(target.velocity);
|
||||
teleported.setHeadYaw(target.yaw);
|
||||
|
||||
return teleported;
|
||||
// Fast path for teleporting within the same dimension.
|
||||
if (teleported.getWorld() == target.newDimension()) {
|
||||
if (teleported instanceof ServerPlayerEntity serverPlayerEntity) {
|
||||
serverPlayerEntity.networkHandler.requestTeleport(target.pos().x, target.pos().y, target.pos().z, target.yaw(), target.pitch());
|
||||
} else {
|
||||
teleported.refreshPositionAndAngles(target.pos().x, target.pos().y, target.pos().z, target.yaw(), target.pitch());
|
||||
}
|
||||
|
||||
return (E) teleported.moveToWorld(dimension);
|
||||
} finally {
|
||||
((Teleportable) teleported).fabric_setCustomTeleportTarget(null);
|
||||
teleported.setVelocity(target.velocity());
|
||||
teleported.setHeadYaw(target.yaw());
|
||||
|
||||
return teleported;
|
||||
}
|
||||
|
||||
return (E) teleported.moveToWorld(() -> target);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,28 +0,0 @@
|
|||
/*
|
||||
* Copyright (c) 2016, 2017, 2018, 2019 FabricMC
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
package net.fabricmc.fabric.impl.dimension;
|
||||
|
||||
import org.jetbrains.annotations.Nullable;
|
||||
|
||||
import net.minecraft.world.TeleportTarget;
|
||||
|
||||
public interface Teleportable {
|
||||
/**
|
||||
* Sets the last target set when a user of the API requested teleportation, or null.
|
||||
*/
|
||||
void fabric_setCustomTeleportTarget(@Nullable TeleportTarget teleportTarget);
|
||||
}
|
|
@ -1,74 +0,0 @@
|
|||
/*
|
||||
* Copyright (c) 2016, 2017, 2018, 2019 FabricMC
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
package net.fabricmc.fabric.mixin.dimension;
|
||||
|
||||
import org.jetbrains.annotations.Nullable;
|
||||
import org.spongepowered.asm.mixin.Mixin;
|
||||
import org.spongepowered.asm.mixin.Unique;
|
||||
import org.spongepowered.asm.mixin.injection.At;
|
||||
import org.spongepowered.asm.mixin.injection.Inject;
|
||||
import org.spongepowered.asm.mixin.injection.Redirect;
|
||||
import org.spongepowered.asm.mixin.injection.callback.CallbackInfoReturnable;
|
||||
|
||||
import net.minecraft.entity.Entity;
|
||||
import net.minecraft.registry.RegistryKey;
|
||||
import net.minecraft.server.network.ServerPlayerEntity;
|
||||
import net.minecraft.server.world.ServerWorld;
|
||||
import net.minecraft.world.TeleportTarget;
|
||||
import net.minecraft.world.World;
|
||||
|
||||
import net.fabricmc.fabric.impl.dimension.Teleportable;
|
||||
|
||||
/**
|
||||
* This mixin implements {@link Entity#getTeleportTarget(ServerWorld)} for modded dimensions, as Vanilla will
|
||||
* not return a teleport target for anything but Vanilla dimensions and prevents changing teleport target in
|
||||
* {@link ServerPlayerEntity#getTeleportTarget(ServerWorld)} when teleporting to END using api.
|
||||
* This also prevents several End dimension-specific code when teleporting using api.
|
||||
*/
|
||||
@Mixin(value = {ServerPlayerEntity.class, Entity.class})
|
||||
public class EntityMixin implements Teleportable {
|
||||
@Unique
|
||||
@Nullable
|
||||
protected TeleportTarget customTeleportTarget;
|
||||
|
||||
@Override
|
||||
public void fabric_setCustomTeleportTarget(TeleportTarget teleportTarget) {
|
||||
this.customTeleportTarget = teleportTarget;
|
||||
}
|
||||
|
||||
@Inject(method = "getTeleportTarget", at = @At("HEAD"), cancellable = true, allow = 1)
|
||||
public void getTeleportTarget(ServerWorld destination, CallbackInfoReturnable<TeleportTarget> cir) {
|
||||
// Check if a destination has been set for the entity currently being teleported
|
||||
TeleportTarget customTarget = this.customTeleportTarget;
|
||||
|
||||
if (customTarget != null) {
|
||||
cir.setReturnValue(customTarget);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* This stops the following behaviors, in 1 mixin.
|
||||
* - ServerWorld#createEndSpawnPlatform in Entity
|
||||
* - End-to-overworld spawning behavior in ServerPlayerEntity
|
||||
* - ServerPlayerEntity#createEndSpawnPlatform in ServerPlayerEntity
|
||||
*/
|
||||
@Redirect(method = "moveToWorld", at = @At(value = "FIELD", target = "Lnet/minecraft/world/World;END:Lnet/minecraft/registry/RegistryKey;"))
|
||||
private RegistryKey<World> stopEndSpecificBehavior() {
|
||||
if (this.customTeleportTarget != null) return null;
|
||||
return World.END;
|
||||
}
|
||||
}
|
|
@ -3,7 +3,6 @@
|
|||
"package": "net.fabricmc.fabric.mixin.dimension",
|
||||
"compatibilityLevel": "JAVA_17",
|
||||
"mixins": [
|
||||
"EntityMixin",
|
||||
"DimensionOptionsRegistryHolderMixin",
|
||||
"Schema2832Mixin",
|
||||
"TaggedChoiceMixin",
|
||||
|
|
|
@ -78,15 +78,15 @@ public class FabricDimensionTest implements ModInitializer {
|
|||
if (entity == null) throw new AssertionError("Could not create entity!");
|
||||
if (!entity.getWorld().getRegistryKey().equals(World.OVERWORLD)) throw new AssertionError("Entity starting world isn't the overworld");
|
||||
|
||||
TeleportTarget target = new TeleportTarget(Vec3d.ZERO, new Vec3d(1, 1, 1), 45f, 60f);
|
||||
TeleportTarget target = new TeleportTarget(world, Vec3d.ZERO, new Vec3d(1, 1, 1), 45f, 60f);
|
||||
|
||||
Entity teleported = FabricDimensions.teleport(entity, world, target);
|
||||
Entity teleported = FabricDimensions.teleport(entity, target);
|
||||
|
||||
if (teleported == null) throw new AssertionError("Entity didn't teleport");
|
||||
|
||||
if (!teleported.getWorld().getRegistryKey().equals(WORLD_KEY)) throw new AssertionError("Target world not reached.");
|
||||
|
||||
if (!teleported.getPos().equals(target.position)) throw new AssertionError("Target Position not reached.");
|
||||
if (!teleported.getPos().equals(target.pos())) throw new AssertionError("Target Position not reached.");
|
||||
});
|
||||
|
||||
CommandRegistrationCallback.EVENT.register((dispatcher, registryAccess, environment) -> {
|
||||
|
@ -124,8 +124,8 @@ public class FabricDimensionTest implements ModInitializer {
|
|||
ServerWorld modWorld = getModWorld(context);
|
||||
|
||||
if (serverWorld != modWorld) {
|
||||
TeleportTarget target = new TeleportTarget(new Vec3d(0.5, 101, 0.5), Vec3d.ZERO, 0, 0);
|
||||
FabricDimensions.teleport(player, modWorld, target);
|
||||
TeleportTarget target = new TeleportTarget(modWorld, new Vec3d(0.5, 101, 0.5), Vec3d.ZERO, 0, 0);
|
||||
FabricDimensions.teleport(player, target);
|
||||
|
||||
if (player.getWorld() != modWorld) {
|
||||
throw FAILED_EXCEPTION.create();
|
||||
|
@ -134,9 +134,9 @@ public class FabricDimensionTest implements ModInitializer {
|
|||
modWorld.setBlockState(new BlockPos(0, 100, 0), Blocks.DIAMOND_BLOCK.getDefaultState());
|
||||
modWorld.setBlockState(new BlockPos(0, 101, 0), Blocks.TORCH.getDefaultState());
|
||||
} else {
|
||||
TeleportTarget target = new TeleportTarget(new Vec3d(0, 100, 0), Vec3d.ZERO,
|
||||
TeleportTarget target = new TeleportTarget(getWorld(context, World.OVERWORLD), new Vec3d(0, 100, 0), Vec3d.ZERO,
|
||||
(float) Math.random() * 360 - 180, (float) Math.random() * 360 - 180);
|
||||
FabricDimensions.teleport(player, getWorld(context, World.OVERWORLD), target);
|
||||
FabricDimensions.teleport(player, target);
|
||||
}
|
||||
|
||||
return 1;
|
||||
|
@ -150,8 +150,8 @@ public class FabricDimensionTest implements ModInitializer {
|
|||
return 1;
|
||||
}
|
||||
|
||||
TeleportTarget target = new TeleportTarget(player.getPos().add(5, 0, 0), player.getVelocity(), player.getYaw(), player.getPitch());
|
||||
FabricDimensions.teleport(player, (ServerWorld) player.getWorld(), target);
|
||||
TeleportTarget target = new TeleportTarget((ServerWorld) player.getWorld(), player.getPos().add(5, 0, 0), player.getVelocity(), player.getYaw(), player.getPitch());
|
||||
FabricDimensions.teleport(player, target);
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
@ -175,16 +175,16 @@ public class FabricDimensionTest implements ModInitializer {
|
|||
return 1;
|
||||
}
|
||||
|
||||
TeleportTarget target = new TeleportTarget(player.getPos(), player.getVelocity(), player.getYaw(), player.getPitch());
|
||||
FabricDimensions.teleport(entity, (ServerWorld) entity.getWorld(), target);
|
||||
TeleportTarget target = new TeleportTarget((ServerWorld) entity.getWorld(), player.getPos(), player.getVelocity(), player.getYaw(), player.getPitch());
|
||||
FabricDimensions.teleport(entity, target);
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
private int testVanillaTeleport(CommandContext<ServerCommandSource> context, ServerWorld targetWorld) throws CommandSyntaxException {
|
||||
Entity entity = context.getSource().getEntityOrThrow();
|
||||
TeleportTarget target = new TeleportTarget(entity.getPos(), entity.getVelocity(), entity.getYaw(), entity.getPitch());
|
||||
FabricDimensions.teleport(entity, targetWorld, target);
|
||||
TeleportTarget target = new TeleportTarget(targetWorld, entity.getPos(), entity.getVelocity(), entity.getYaw(), entity.getPitch());
|
||||
FabricDimensions.teleport(entity, target);
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
|
|
@ -38,7 +38,7 @@ abstract class EntityMixin {
|
|||
public World world;
|
||||
|
||||
@Inject(method = "moveToWorld", at = @At("RETURN"))
|
||||
private void afterWorldChanged(ServerWorld destination, CallbackInfoReturnable<Entity> cir) {
|
||||
private void afterWorldChanged(Entity.TeleportTargetSupplier targetSupplier, CallbackInfoReturnable<Entity> cir) {
|
||||
// Ret will only have an entity if the teleport worked (entity not removed, teleportTarget was valid, entity was successfully created)
|
||||
Entity ret = cir.getReturnValue();
|
||||
|
||||
|
|
|
@ -21,6 +21,7 @@ import org.spongepowered.asm.mixin.injection.At;
|
|||
import org.spongepowered.asm.mixin.injection.Inject;
|
||||
import org.spongepowered.asm.mixin.injection.callback.CallbackInfoReturnable;
|
||||
|
||||
import net.minecraft.entity.Entity;
|
||||
import net.minecraft.server.PlayerManager;
|
||||
import net.minecraft.server.network.ServerPlayerEntity;
|
||||
|
||||
|
@ -29,7 +30,7 @@ import net.fabricmc.fabric.api.entity.event.v1.ServerPlayerEvents;
|
|||
@Mixin(PlayerManager.class)
|
||||
abstract class PlayerManagerMixin {
|
||||
@Inject(method = "respawnPlayer", at = @At("TAIL"))
|
||||
private void afterRespawn(ServerPlayerEntity oldPlayer, boolean alive, CallbackInfoReturnable<ServerPlayerEntity> cir) {
|
||||
private void afterRespawn(ServerPlayerEntity oldPlayer, boolean alive, Entity.RemovalReason removalReason, CallbackInfoReturnable<ServerPlayerEntity> cir) {
|
||||
ServerPlayerEvents.AFTER_RESPAWN.invoker().afterRespawn(oldPlayer, cir.getReturnValue(), alive);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -40,7 +40,6 @@ import net.minecraft.network.packet.c2s.play.PlayerActionC2SPacket;
|
|||
import net.minecraft.network.packet.c2s.play.PlayerInteractBlockC2SPacket;
|
||||
import net.minecraft.network.packet.c2s.play.PlayerInteractEntityC2SPacket;
|
||||
import net.minecraft.network.packet.c2s.play.PlayerInteractItemC2SPacket;
|
||||
import net.minecraft.network.packet.c2s.play.PlayerMoveC2SPacket;
|
||||
import net.minecraft.util.ActionResult;
|
||||
import net.minecraft.util.Hand;
|
||||
import net.minecraft.util.TypedActionResult;
|
||||
|
@ -118,7 +117,7 @@ public abstract class ClientPlayerInteractionManagerMixin {
|
|||
}
|
||||
}
|
||||
|
||||
@Inject(at = @At(value = "INVOKE", target = "Lnet/minecraft/client/network/ClientPlayNetworkHandler;sendPacket(Lnet/minecraft/network/packet/Packet;)V", ordinal = 0), method = "interactItem", cancellable = true)
|
||||
@Inject(at = @At(value = "INVOKE", target = "Lnet/minecraft/client/network/ClientPlayerInteractionManager;syncSelectedSlot()V", ordinal = 0), method = "interactItem", cancellable = true)
|
||||
public void interactItem(PlayerEntity player, Hand hand, CallbackInfoReturnable<ActionResult> info) {
|
||||
// hook interactBlock between the spectator check and sending the first packet to invoke the use item event first
|
||||
// this needs to be in interactBlock to avoid sending a packet in line with the event javadoc
|
||||
|
@ -126,10 +125,8 @@ public abstract class ClientPlayerInteractionManagerMixin {
|
|||
|
||||
if (result.getResult() != ActionResult.PASS) {
|
||||
if (result.getResult() == ActionResult.SUCCESS) {
|
||||
// send the move packet like vanilla to ensure the position+view vectors are accurate
|
||||
networkHandler.sendPacket(new PlayerMoveC2SPacket.Full(player.getX(), player.getY(), player.getZ(), player.getYaw(), player.getPitch(), player.isOnGround()));
|
||||
// send interaction packet to the server with a new sequentially assigned id
|
||||
sendSequencedPacket((ClientWorld) player.getWorld(), id -> new PlayerInteractItemC2SPacket(hand, id));
|
||||
sendSequencedPacket((ClientWorld) player.getWorld(), id -> new PlayerInteractItemC2SPacket(hand, id, player.getYaw(), player.getPitch()));
|
||||
}
|
||||
|
||||
info.setReturnValue(result.getResult());
|
||||
|
|
|
@ -16,6 +16,8 @@
|
|||
|
||||
package net.fabricmc.fabric.mixin.item;
|
||||
|
||||
import java.util.function.Consumer;
|
||||
|
||||
import com.llamalad7.mixinextras.injector.wrapoperation.Operation;
|
||||
import com.llamalad7.mixinextras.injector.wrapoperation.WrapOperation;
|
||||
import com.llamalad7.mixinextras.sugar.Local;
|
||||
|
@ -36,10 +38,14 @@ import net.fabricmc.fabric.impl.item.ItemExtensions;
|
|||
|
||||
@Mixin(ItemStack.class)
|
||||
public abstract class ItemStackMixin implements FabricItemStack {
|
||||
@Shadow public abstract Item getItem();
|
||||
@Shadow
|
||||
public abstract Item getItem();
|
||||
|
||||
@WrapOperation(method = "damage(ILnet/minecraft/entity/LivingEntity;Lnet/minecraft/entity/EquipmentSlot;)V", at = @At(value = "INVOKE", target = "Lnet/minecraft/item/ItemStack;damage(ILnet/minecraft/server/world/ServerWorld;Lnet/minecraft/server/network/ServerPlayerEntity;Ljava/lang/Runnable;)V"))
|
||||
private void hookDamage(ItemStack instance, int amount, ServerWorld serverWorld, ServerPlayerEntity serverPlayerEntity, Runnable runnable, Operation<Void> original, @Local(argsOnly = true) EquipmentSlot slot) {
|
||||
@Shadow
|
||||
public abstract void decrement(int amount);
|
||||
|
||||
@WrapOperation(method = "damage(ILnet/minecraft/entity/LivingEntity;Lnet/minecraft/entity/EquipmentSlot;)V", at = @At(value = "INVOKE", target = "Lnet/minecraft/item/ItemStack;damage(ILnet/minecraft/server/world/ServerWorld;Lnet/minecraft/server/network/ServerPlayerEntity;Ljava/util/function/Consumer;)V"))
|
||||
private void hookDamage(ItemStack instance, int amount, ServerWorld serverWorld, ServerPlayerEntity serverPlayerEntity, Consumer<Item> consumer, Operation<Void> original, @Local(argsOnly = true) EquipmentSlot slot) {
|
||||
CustomDamageHandler handler = ((ItemExtensions) getItem()).fabric_getCustomDamageHandler();
|
||||
|
||||
if (handler != null) {
|
||||
|
@ -47,13 +53,14 @@ public abstract class ItemStackMixin implements FabricItemStack {
|
|||
MutableBoolean mut = new MutableBoolean(false);
|
||||
amount = handler.damage((ItemStack) (Object) this, amount, serverPlayerEntity, slot, () -> {
|
||||
mut.setTrue();
|
||||
runnable.run();
|
||||
this.decrement(1);
|
||||
consumer.accept(this.getItem());
|
||||
});
|
||||
|
||||
// If item is broken, there's no reason to call the original.
|
||||
if (mut.booleanValue()) return;
|
||||
}
|
||||
|
||||
original.call(instance, amount, serverWorld, serverPlayerEntity, runnable);
|
||||
original.call(instance, amount, serverWorld, serverPlayerEntity, consumer);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -31,7 +31,7 @@ import net.fabricmc.fabric.impl.item.ItemExtensions;
|
|||
@Mixin(LivingEntity.class)
|
||||
abstract class LivingEntityMixin {
|
||||
@Inject(method = "getPreferredEquipmentSlot", at = @At(value = "HEAD"), cancellable = true)
|
||||
private static void onGetPreferredEquipmentSlot(ItemStack stack, CallbackInfoReturnable<EquipmentSlot> info) {
|
||||
private void onGetPreferredEquipmentSlot(ItemStack stack, CallbackInfoReturnable<EquipmentSlot> info) {
|
||||
EquipmentSlotProvider equipmentSlotProvider = ((ItemExtensions) stack.getItem()).fabric_getEquipmentSlotProvider();
|
||||
|
||||
if (equipmentSlotProvider != null) {
|
||||
|
|
|
@ -21,7 +21,7 @@ import org.spongepowered.asm.mixin.injection.At;
|
|||
import org.spongepowered.asm.mixin.injection.Inject;
|
||||
import org.spongepowered.asm.mixin.injection.callback.CallbackInfoReturnable;
|
||||
|
||||
import net.minecraft.class_9761;
|
||||
import net.minecraft.world.chunk.AbstractChunkHolder;
|
||||
import net.minecraft.world.chunk.Chunk;
|
||||
import net.minecraft.world.chunk.ChunkGenerating;
|
||||
import net.minecraft.world.chunk.ChunkGenerationContext;
|
||||
|
@ -32,7 +32,7 @@ import net.fabricmc.fabric.api.event.lifecycle.v1.ServerChunkEvents;
|
|||
@Mixin(ChunkGenerating.class)
|
||||
abstract class ChunkGeneratingMixin {
|
||||
@Inject(method = "method_60553", at = @At("TAIL"))
|
||||
private static void onChunkLoad(Chunk chunk, ChunkGenerationContext chunkGenerationContext, class_9761 arg, CallbackInfoReturnable<Chunk> callbackInfoReturnable) {
|
||||
private static void onChunkLoad(Chunk chunk, ChunkGenerationContext chunkGenerationContext, AbstractChunkHolder chunkHolder, CallbackInfoReturnable<Chunk> callbackInfoReturnable) {
|
||||
// We fire the event at TAIL since the chunk is guaranteed to be a WorldChunk then.
|
||||
ServerChunkEvents.CHUNK_LOAD.invoker().onChunkLoad(chunkGenerationContext.world(), (WorldChunk) callbackInfoReturnable.getReturnValue());
|
||||
}
|
||||
|
|
|
@ -24,14 +24,14 @@ import org.spongepowered.asm.mixin.injection.Inject;
|
|||
import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
|
||||
|
||||
import net.minecraft.server.world.ChunkHolder;
|
||||
import net.minecraft.server.world.ServerChunkLoadingManager;
|
||||
import net.minecraft.server.world.ServerWorld;
|
||||
import net.minecraft.server.world.ThreadedAnvilChunkStorage;
|
||||
import net.minecraft.world.chunk.WorldChunk;
|
||||
|
||||
import net.fabricmc.fabric.api.event.lifecycle.v1.ServerChunkEvents;
|
||||
|
||||
@Mixin(ThreadedAnvilChunkStorage.class)
|
||||
public abstract class ThreadedAnvilChunkStorageMixin {
|
||||
@Mixin(ServerChunkLoadingManager.class)
|
||||
public abstract class ServerChunkLoadingManagerMixin {
|
||||
@Shadow
|
||||
@Final
|
||||
private ServerWorld world;
|
||||
|
@ -45,6 +45,6 @@ public abstract class ThreadedAnvilChunkStorageMixin {
|
|||
*/
|
||||
@Inject(method = "method_60440", at = @At(value = "INVOKE", target = "Lnet/minecraft/world/chunk/WorldChunk;setLoadedToWorld(Z)V", shift = At.Shift.AFTER))
|
||||
private void onChunkUnload(ChunkHolder chunkHolder, long l, CallbackInfo ci) {
|
||||
ServerChunkEvents.CHUNK_UNLOAD.invoker().onChunkUnload(this.world, (WorldChunk) chunkHolder.method_60471());
|
||||
ServerChunkEvents.CHUNK_UNLOAD.invoker().onChunkUnload(this.world, (WorldChunk) chunkHolder.getLatest());
|
||||
}
|
||||
}
|
|
@ -10,7 +10,7 @@
|
|||
"PlayerManagerMixin",
|
||||
"ServerWorldServerEntityHandlerMixin",
|
||||
"ServerWorldMixin",
|
||||
"ThreadedAnvilChunkStorageMixin",
|
||||
"ServerChunkLoadingManagerMixin",
|
||||
"WorldMixin"
|
||||
],
|
||||
"server": [
|
||||
|
|
|
@ -26,9 +26,9 @@ import net.minecraft.entity.Entity;
|
|||
import net.minecraft.server.MinecraftServer;
|
||||
import net.minecraft.server.network.PlayerAssociatedNetworkHandler;
|
||||
import net.minecraft.server.network.ServerPlayerEntity;
|
||||
import net.minecraft.server.world.ServerChunkLoadingManager;
|
||||
import net.minecraft.server.world.ServerChunkManager;
|
||||
import net.minecraft.server.world.ServerWorld;
|
||||
import net.minecraft.server.world.ThreadedAnvilChunkStorage;
|
||||
import net.minecraft.util.math.BlockPos;
|
||||
import net.minecraft.util.math.ChunkPos;
|
||||
import net.minecraft.util.math.Vec3d;
|
||||
|
@ -36,7 +36,7 @@ import net.minecraft.util.math.Vec3i;
|
|||
import net.minecraft.world.chunk.ChunkManager;
|
||||
|
||||
import net.fabricmc.fabric.mixin.networking.accessor.EntityTrackerAccessor;
|
||||
import net.fabricmc.fabric.mixin.networking.accessor.ThreadedAnvilChunkStorageAccessor;
|
||||
import net.fabricmc.fabric.mixin.networking.accessor.ServerChunkLoadingManagerAccessor;
|
||||
|
||||
/**
|
||||
* Helper methods to lookup players in a server.
|
||||
|
@ -91,7 +91,7 @@ public final class PlayerLookup {
|
|||
Objects.requireNonNull(world, "The world cannot be null");
|
||||
Objects.requireNonNull(pos, "The chunk pos cannot be null");
|
||||
|
||||
return world.getChunkManager().threadedAnvilChunkStorage.getPlayersWatchingChunk(pos, false);
|
||||
return world.getChunkManager().chunkLoadingManager.getPlayersWatchingChunk(pos, false);
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -112,8 +112,8 @@ public final class PlayerLookup {
|
|||
ChunkManager manager = entity.getWorld().getChunkManager();
|
||||
|
||||
if (manager instanceof ServerChunkManager) {
|
||||
ThreadedAnvilChunkStorage storage = ((ServerChunkManager) manager).threadedAnvilChunkStorage;
|
||||
EntityTrackerAccessor tracker = ((ThreadedAnvilChunkStorageAccessor) storage).getEntityTrackers().get(entity.getId());
|
||||
ServerChunkLoadingManager chunkLoadingManager = ((ServerChunkManager) manager).chunkLoadingManager;
|
||||
EntityTrackerAccessor tracker = ((ServerChunkLoadingManagerAccessor) chunkLoadingManager).getEntityTrackers().get(entity.getId());
|
||||
|
||||
// return an immutable collection to guard against accidental removals.
|
||||
if (tracker != null) {
|
||||
|
|
|
@ -23,7 +23,7 @@ import org.spongepowered.asm.mixin.gen.Accessor;
|
|||
|
||||
import net.minecraft.server.network.PlayerAssociatedNetworkHandler;
|
||||
|
||||
@Mixin(targets = "net/minecraft/server/world/ThreadedAnvilChunkStorage$EntityTracker")
|
||||
@Mixin(targets = "net/minecraft/server/world/ServerChunkLoadingManager$EntityTracker")
|
||||
public interface EntityTrackerAccessor {
|
||||
@Accessor("listeners")
|
||||
Set<PlayerAssociatedNetworkHandler> getPlayersTracking();
|
||||
|
|
|
@ -20,10 +20,10 @@ import it.unimi.dsi.fastutil.ints.Int2ObjectMap;
|
|||
import org.spongepowered.asm.mixin.Mixin;
|
||||
import org.spongepowered.asm.mixin.gen.Accessor;
|
||||
|
||||
import net.minecraft.server.world.ThreadedAnvilChunkStorage;
|
||||
import net.minecraft.server.world.ServerChunkLoadingManager;
|
||||
|
||||
@Mixin(ThreadedAnvilChunkStorage.class)
|
||||
public interface ThreadedAnvilChunkStorageAccessor {
|
||||
@Mixin(ServerChunkLoadingManager.class)
|
||||
public interface ServerChunkLoadingManagerAccessor {
|
||||
@Accessor
|
||||
Int2ObjectMap<EntityTrackerAccessor> getEntityTrackers();
|
||||
}
|
|
@ -20,7 +20,7 @@
|
|||
"accessor.EntityTrackerAccessor",
|
||||
"accessor.ServerCommonNetworkHandlerAccessor",
|
||||
"accessor.ServerLoginNetworkHandlerAccessor",
|
||||
"accessor.ThreadedAnvilChunkStorageAccessor"
|
||||
"accessor.ServerChunkLoadingManagerAccessor"
|
||||
],
|
||||
"injectors": {
|
||||
"defaultRequire": 1
|
||||
|
|
|
@ -23,7 +23,7 @@ import net.minecraft.block.AbstractBlock;
|
|||
import net.minecraft.block.Block;
|
||||
import net.minecraft.block.BlockState;
|
||||
import net.minecraft.block.MapColor;
|
||||
import net.minecraft.block.enums.Instrument;
|
||||
import net.minecraft.block.enums.NoteBlockInstrument;
|
||||
import net.minecraft.block.piston.PistonBehavior;
|
||||
import net.minecraft.entity.EntityType;
|
||||
import net.minecraft.loot.LootTable;
|
||||
|
@ -375,7 +375,7 @@ public class FabricBlockSettings extends AbstractBlock.Settings {
|
|||
|
||||
@Deprecated
|
||||
@Override
|
||||
public FabricBlockSettings instrument(Instrument instrument) {
|
||||
public FabricBlockSettings instrument(NoteBlockInstrument instrument) {
|
||||
super.instrument(instrument);
|
||||
return this;
|
||||
}
|
||||
|
|
|
@ -26,7 +26,7 @@ import org.spongepowered.asm.mixin.gen.Accessor;
|
|||
import net.minecraft.block.AbstractBlock;
|
||||
import net.minecraft.block.BlockState;
|
||||
import net.minecraft.block.MapColor;
|
||||
import net.minecraft.block.enums.Instrument;
|
||||
import net.minecraft.block.enums.NoteBlockInstrument;
|
||||
import net.minecraft.block.piston.PistonBehavior;
|
||||
import net.minecraft.entity.EntityType;
|
||||
import net.minecraft.loot.LootTable;
|
||||
|
@ -125,7 +125,7 @@ public interface AbstractBlockSettingsAccessor {
|
|||
PistonBehavior getPistonBehavior();
|
||||
|
||||
@Accessor
|
||||
Instrument getInstrument();
|
||||
NoteBlockInstrument getInstrument();
|
||||
|
||||
@Accessor
|
||||
boolean getReplaceable();
|
||||
|
|
|
@ -17,6 +17,7 @@
|
|||
package net.fabricmc.fabric.api.client.rendering.v1;
|
||||
|
||||
import net.minecraft.client.gui.DrawContext;
|
||||
import net.minecraft.client.render.RenderTickCounter;
|
||||
|
||||
import net.fabricmc.fabric.api.event.Event;
|
||||
import net.fabricmc.fabric.api.event.EventFactory;
|
||||
|
@ -32,7 +33,7 @@ public interface HudRenderCallback {
|
|||
* Called after rendering the whole hud, which is displayed in game, in a world.
|
||||
*
|
||||
* @param drawContext the {@link DrawContext} instance
|
||||
* @param tickDelta Progress for linearly interpolating between the previous and current game state
|
||||
* @param tickCounter the {@link RenderTickCounter} instance
|
||||
*/
|
||||
void onHudRender(DrawContext drawContext, float tickDelta);
|
||||
void onHudRender(DrawContext drawContext, RenderTickCounter tickCounter);
|
||||
}
|
||||
|
|
|
@ -24,6 +24,7 @@ import net.minecraft.client.render.Camera;
|
|||
import net.minecraft.client.render.Frustum;
|
||||
import net.minecraft.client.render.GameRenderer;
|
||||
import net.minecraft.client.render.LightmapTextureManager;
|
||||
import net.minecraft.client.render.RenderTickCounter;
|
||||
import net.minecraft.client.render.VertexConsumer;
|
||||
import net.minecraft.client.render.VertexConsumerProvider;
|
||||
import net.minecraft.client.render.WorldRenderer;
|
||||
|
@ -51,9 +52,7 @@ public interface WorldRenderContext {
|
|||
@Nullable
|
||||
MatrixStack matrixStack();
|
||||
|
||||
float tickDelta();
|
||||
|
||||
long limitTime();
|
||||
RenderTickCounter tickCounter();
|
||||
|
||||
boolean blockOutlines();
|
||||
|
||||
|
|
|
@ -24,6 +24,7 @@ import net.minecraft.client.render.Frustum;
|
|||
import net.minecraft.client.render.GameRenderer;
|
||||
import net.minecraft.client.render.LightmapTextureManager;
|
||||
import net.minecraft.client.render.RenderLayer;
|
||||
import net.minecraft.client.render.RenderTickCounter;
|
||||
import net.minecraft.client.render.VertexConsumer;
|
||||
import net.minecraft.client.render.VertexConsumerProvider;
|
||||
import net.minecraft.client.render.WorldRenderer;
|
||||
|
@ -37,9 +38,8 @@ import net.fabricmc.fabric.api.client.rendering.v1.WorldRenderContext;
|
|||
|
||||
public final class WorldRenderContextImpl implements WorldRenderContext.BlockOutlineContext, WorldRenderContext {
|
||||
private WorldRenderer worldRenderer;
|
||||
private RenderTickCounter tickCounter;
|
||||
private MatrixStack matrixStack;
|
||||
private float tickDelta;
|
||||
private long limitTime;
|
||||
private boolean blockOutlines;
|
||||
private Camera camera;
|
||||
private Frustum frustum;
|
||||
|
@ -63,8 +63,7 @@ public final class WorldRenderContextImpl implements WorldRenderContext.BlockOut
|
|||
|
||||
public void prepare(
|
||||
WorldRenderer worldRenderer,
|
||||
float tickDelta,
|
||||
long limitTime,
|
||||
RenderTickCounter delta,
|
||||
boolean blockOutlines,
|
||||
Camera camera,
|
||||
GameRenderer gameRenderer,
|
||||
|
@ -77,9 +76,8 @@ public final class WorldRenderContextImpl implements WorldRenderContext.BlockOut
|
|||
ClientWorld world
|
||||
) {
|
||||
this.worldRenderer = worldRenderer;
|
||||
this.tickCounter = delta;
|
||||
this.matrixStack = null;
|
||||
this.tickDelta = tickDelta;
|
||||
this.limitTime = limitTime;
|
||||
this.blockOutlines = blockOutlines;
|
||||
this.camera = camera;
|
||||
this.gameRenderer = gameRenderer;
|
||||
|
@ -127,13 +125,8 @@ public final class WorldRenderContextImpl implements WorldRenderContext.BlockOut
|
|||
}
|
||||
|
||||
@Override
|
||||
public float tickDelta() {
|
||||
return tickDelta;
|
||||
}
|
||||
|
||||
@Override
|
||||
public long limitTime() {
|
||||
return limitTime;
|
||||
public RenderTickCounter tickCounter() {
|
||||
return this.tickCounter;
|
||||
}
|
||||
|
||||
@Override
|
||||
|
|
|
@ -19,18 +19,18 @@ package net.fabricmc.fabric.mixin.client.rendering;
|
|||
import org.spongepowered.asm.mixin.Mixin;
|
||||
import org.spongepowered.asm.mixin.injection.At;
|
||||
import org.spongepowered.asm.mixin.injection.Inject;
|
||||
import org.spongepowered.asm.mixin.injection.Slice;
|
||||
import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
|
||||
|
||||
import net.minecraft.client.gui.DrawContext;
|
||||
import net.minecraft.client.gui.hud.InGameHud;
|
||||
import net.minecraft.client.render.RenderTickCounter;
|
||||
|
||||
import net.fabricmc.fabric.api.client.rendering.v1.HudRenderCallback;
|
||||
|
||||
@Mixin(InGameHud.class)
|
||||
public class InGameHudMixin {
|
||||
@Inject(method = "render", at = @At(value = "TAIL"), slice = @Slice(from = @At(value = "INVOKE", target = "Lnet/minecraft/client/gui/LayeredDrawer;render(Lnet/minecraft/client/gui/DrawContext;F)V")))
|
||||
public void render(DrawContext drawContext, float tickDelta, CallbackInfo callbackInfo) {
|
||||
HudRenderCallback.EVENT.invoker().onHudRender(drawContext, tickDelta);
|
||||
@Inject(method = "render", at = @At(value = "TAIL"))
|
||||
public void render(DrawContext drawContext, RenderTickCounter tickCounter, CallbackInfo callbackInfo) {
|
||||
HudRenderCallback.EVENT.invoker().onHudRender(drawContext, tickCounter);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -36,6 +36,7 @@ import net.minecraft.client.render.Frustum;
|
|||
import net.minecraft.client.render.GameRenderer;
|
||||
import net.minecraft.client.render.LightmapTextureManager;
|
||||
import net.minecraft.client.render.RenderLayer;
|
||||
import net.minecraft.client.render.RenderTickCounter;
|
||||
import net.minecraft.client.render.VertexConsumer;
|
||||
import net.minecraft.client.render.WorldRenderer;
|
||||
import net.minecraft.client.util.math.MatrixStack;
|
||||
|
@ -61,8 +62,8 @@ public abstract class WorldRendererMixin {
|
|||
@Unique private final WorldRenderContextImpl context = new WorldRenderContextImpl();
|
||||
|
||||
@Inject(method = "render", at = @At("HEAD"))
|
||||
private void beforeRender(float tickDelta, long limitTime, boolean renderBlockOutline, Camera camera, GameRenderer gameRenderer, LightmapTextureManager lightmapTextureManager, Matrix4f positionMatrix, Matrix4f projectionMatrix, CallbackInfo ci) {
|
||||
context.prepare((WorldRenderer) (Object) this, tickDelta, limitTime, renderBlockOutline, camera, gameRenderer, lightmapTextureManager, projectionMatrix, positionMatrix, bufferBuilders.getEntityVertexConsumers(), world.getProfiler(), transparencyPostProcessor != null, world);
|
||||
private void beforeRender(RenderTickCounter tickCounter, boolean renderBlockOutline, Camera camera, GameRenderer gameRenderer, LightmapTextureManager lightmapTextureManager, Matrix4f positionMatrix, Matrix4f projectionMatrix, CallbackInfo ci) {
|
||||
context.prepare((WorldRenderer) (Object) this, tickCounter, renderBlockOutline, camera, gameRenderer, lightmapTextureManager, projectionMatrix, positionMatrix, bufferBuilders.getEntityVertexConsumers(), world.getProfiler(), transparencyPostProcessor != null, world);
|
||||
WorldRenderEvents.START.invoker().onStart(context);
|
||||
}
|
||||
|
||||
|
|
|
@ -2,9 +2,9 @@ org.gradle.jvmargs=-Xmx2560M
|
|||
org.gradle.parallel=true
|
||||
fabric.loom.multiProjectOptimisation=true
|
||||
|
||||
version=0.98.1
|
||||
minecraft_version=24w19b
|
||||
yarn_version=+build.1
|
||||
version=0.98.2
|
||||
minecraft_version=24w20a
|
||||
yarn_version=+build.3
|
||||
loader_version=0.15.11
|
||||
installer_version=1.0.1
|
||||
|
||||
|
@ -13,7 +13,7 @@ curseforge_minecraft_version=1.21-Snapshot
|
|||
|
||||
# Do not manually update, use the bumpversions task:
|
||||
fabric-api-base-version=0.4.41
|
||||
fabric-api-lookup-api-v1-version=1.6.61
|
||||
fabric-api-lookup-api-v1-version=1.6.62
|
||||
fabric-biome-api-v1-version=13.0.26
|
||||
fabric-block-api-v1-version=1.0.21
|
||||
fabric-block-view-api-v2-version=1.0.9
|
||||
|
@ -21,43 +21,43 @@ fabric-blockrenderlayer-v1-version=1.1.51
|
|||
fabric-command-api-v1-version=1.2.46
|
||||
fabric-command-api-v2-version=2.2.25
|
||||
fabric-commands-v0-version=0.2.63
|
||||
fabric-content-registries-v0-version=8.0.7
|
||||
fabric-content-registries-v0-version=8.0.8
|
||||
fabric-crash-report-info-v1-version=0.2.28
|
||||
fabric-data-attachment-api-v1-version=1.1.18
|
||||
fabric-data-generation-api-v1-version=20.1.1
|
||||
fabric-dimensions-v1-version=2.1.70
|
||||
fabric-entity-events-v1-version=1.6.9
|
||||
fabric-events-interaction-v0-version=0.7.8
|
||||
fabric-data-attachment-api-v1-version=1.1.19
|
||||
fabric-data-generation-api-v1-version=20.1.2
|
||||
fabric-dimensions-v1-version=3.0.0
|
||||
fabric-entity-events-v1-version=1.6.10
|
||||
fabric-events-interaction-v0-version=0.7.9
|
||||
fabric-game-rule-api-v1-version=1.0.51
|
||||
fabric-gametest-api-v1-version=1.3.19
|
||||
fabric-item-api-v1-version=9.1.1
|
||||
fabric-item-api-v1-version=9.1.2
|
||||
fabric-item-group-api-v1-version=4.0.41
|
||||
fabric-key-binding-api-v1-version=1.0.46
|
||||
fabric-keybindings-v0-version=0.2.44
|
||||
fabric-lifecycle-events-v1-version=2.3.6
|
||||
fabric-lifecycle-events-v1-version=2.3.7
|
||||
fabric-loot-api-v2-version=3.0.7
|
||||
fabric-message-api-v1-version=6.0.11
|
||||
fabric-model-loading-api-v1-version=1.0.13
|
||||
fabric-models-v0-version=0.4.12
|
||||
fabric-networking-api-v1-version=4.0.9
|
||||
fabric-object-builder-api-v1-version=15.1.6
|
||||
fabric-networking-api-v1-version=4.0.10
|
||||
fabric-object-builder-api-v1-version=15.1.7
|
||||
fabric-particles-v1-version=4.0.1
|
||||
fabric-recipe-api-v1-version=5.0.4
|
||||
fabric-registry-sync-v0-version=5.0.17
|
||||
fabric-recipe-api-v1-version=5.0.5
|
||||
fabric-registry-sync-v0-version=5.0.18
|
||||
fabric-renderer-api-v1-version=3.2.13
|
||||
fabric-renderer-indigo-version=1.5.13
|
||||
fabric-renderer-registries-v1-version=3.2.62
|
||||
fabric-renderer-registries-v1-version=3.2.63
|
||||
fabric-rendering-data-attachment-v1-version=0.3.47
|
||||
fabric-rendering-fluids-v1-version=3.1.4
|
||||
fabric-rendering-v0-version=1.1.65
|
||||
fabric-rendering-v1-version=4.2.5
|
||||
fabric-rendering-v0-version=1.1.66
|
||||
fabric-rendering-v1-version=5.0.0
|
||||
fabric-resource-conditions-api-v1-version=4.1.1
|
||||
fabric-resource-loader-v0-version=1.1.1
|
||||
fabric-screen-api-v1-version=2.0.22
|
||||
fabric-screen-handler-api-v1-version=1.3.74
|
||||
fabric-screen-handler-api-v1-version=1.3.75
|
||||
fabric-sound-api-v1-version=1.0.22
|
||||
fabric-transfer-api-v1-version=5.1.8
|
||||
fabric-transfer-api-v1-version=5.1.9
|
||||
fabric-transitive-access-wideners-v1-version=6.0.11
|
||||
fabric-convention-tags-v1-version=2.0.6
|
||||
fabric-convention-tags-v2-version=2.1.1
|
||||
fabric-convention-tags-v1-version=2.0.7
|
||||
fabric-convention-tags-v2-version=2.1.2
|
||||
fabric-client-tags-api-v1-version=1.1.14
|
||||
|
|
Loading…
Add table
Reference in a new issue