Port to 24w20a ()

* Port to 24w20a

* Update yarn

* Fix TODOs

* Bump version

---------

Co-authored-by: Drex <nicknamedrex@gmail.com>
This commit is contained in:
modmuss 2024-05-16 18:47:15 +01:00 committed by GitHub
parent 5e47b9cbbd
commit e633f8839b
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
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
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
fabric-networking-api-v1/src/main
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
gradle.properties

View file

@ -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);

View file

@ -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);
}
}

View file

@ -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);
}
}

View file

@ -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);
}

View file

@ -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;
}
}

View file

@ -3,7 +3,6 @@
"package": "net.fabricmc.fabric.mixin.dimension",
"compatibilityLevel": "JAVA_17",
"mixins": [
"EntityMixin",
"DimensionOptionsRegistryHolderMixin",
"Schema2832Mixin",
"TaggedChoiceMixin",

View file

@ -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;
}

View file

@ -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();

View file

@ -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);
}
}

View file

@ -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());

View file

@ -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);
}
}

View file

@ -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) {

View file

@ -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());
}

View file

@ -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());
}
}

View file

@ -10,7 +10,7 @@
"PlayerManagerMixin",
"ServerWorldServerEntityHandlerMixin",
"ServerWorldMixin",
"ThreadedAnvilChunkStorageMixin",
"ServerChunkLoadingManagerMixin",
"WorldMixin"
],
"server": [

View file

@ -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) {

View file

@ -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();

View file

@ -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();
}

View file

@ -20,7 +20,7 @@
"accessor.EntityTrackerAccessor",
"accessor.ServerCommonNetworkHandlerAccessor",
"accessor.ServerLoginNetworkHandlerAccessor",
"accessor.ThreadedAnvilChunkStorageAccessor"
"accessor.ServerChunkLoadingManagerAccessor"
],
"injectors": {
"defaultRequire": 1

View file

@ -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;
}

View file

@ -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();

View file

@ -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);
}

View file

@ -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();

View file

@ -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

View file

@ -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);
}
}

View file

@ -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);
}

View file

@ -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