From bbf4c019860817ee523a99a1a0354c05bcf416cc Mon Sep 17 00:00:00 2001
From: Waterpicker <Waterpickerenternity@gmail.com>
Date: Sat, 28 Nov 2020 13:47:10 -0600
Subject: [PATCH] Update the Dimension API (#1172)

* Reimplement Dimension API

* Forgot to update dimension-api mixins.json

* Did suggested changes.

* Move Nullable import.

* Remove unneeded whitespace

* Added automatic testing testmod feature.

* ...Removed unused imports...

* And other stuff the auto style system complains about...

* ...

* Drop Networking

* Completed suggested changes.

* Revert "Completed suggested changes."

This reverts commit 97740e6134400f6e1c826400cb9702b66b39653c.

* Did requested changes.

* Forgot colon

* Turn plural to singular

Co-authored-by: i509VCB <git@i509.me>
---
 fabric-dimensions-v1/build.gradle             |   6 +-
 .../fabric/api/dimension/v1/EntityPlacer.java |  50 ---------
 .../api/dimension/v1/FabricDimensions.java    |  77 +++----------
 .../dimension/FabricDimensionInternals.java   | 105 +++---------------
 .../fabric/mixin/dimension/EntityHooks.java   |  33 ------
 .../{MixinEntity.java => EntityMixin.java}    |  21 +++-
 .../mixin/dimension/MixinPortalForcer.java    |  55 ---------
 .../fabric/mixin/dimension/MixinWorld.java    |  58 ----------
 .../mixin/dimension/ServerBugfixMixin.java    |  92 +++++++++++++++
 .../fabric-dimensions-v1.mixins.json          |   6 +-
 .../test/dimension/FabricDimensionTest.java   |  88 +++++++++++----
 .../test/dimension/VoidChunkGenerator.java    |   4 +-
 .../dimension}/void.json                      |   0
 .../dimension_type}/void_type.json            |   2 +-
 settings.gradle                               |   2 +-
 15 files changed, 216 insertions(+), 383 deletions(-)
 delete mode 100644 fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/api/dimension/v1/EntityPlacer.java
 delete mode 100644 fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/EntityHooks.java
 rename fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/{MixinEntity.java => EntityMixin.java} (58%)
 delete mode 100644 fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/MixinPortalForcer.java
 delete mode 100644 fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/MixinWorld.java
 create mode 100644 fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/ServerBugfixMixin.java
 rename fabric-dimensions-v1/src/testmod/resources/data/{minecraft/dimension/fabric_dimension => fabric_dimension/dimension}/void.json (100%)
 rename fabric-dimensions-v1/src/testmod/resources/data/{minecraft/dimension_type/fabric_dimension => fabric_dimension/dimension_type}/void_type.json (92%)

diff --git a/fabric-dimensions-v1/build.gradle b/fabric-dimensions-v1/build.gradle
index d3533865b..0c147e34a 100644
--- a/fabric-dimensions-v1/build.gradle
+++ b/fabric-dimensions-v1/build.gradle
@@ -3,10 +3,10 @@ version = getSubprojectVersion(project, "1.0.0")
 
 dependencies {
 	testmodCompile project(path: ':fabric-command-api-v1', configuration: 'dev')
+	testmodCompile project(path: ':fabric-resource-loader-v0', configuration: 'dev')
+	testmodCompile project(path: ':fabric-lifecycle-events-v1', configuration: 'dev')
 }
 
 moduleDependencies(project, [
-		'fabric-api-base',
-		'fabric-networking-v0',
-		'fabric-registry-sync-v0'
+		'fabric-api-base'
 ])
diff --git a/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/api/dimension/v1/EntityPlacer.java b/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/api/dimension/v1/EntityPlacer.java
deleted file mode 100644
index 4fb674fcd..000000000
--- a/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/api/dimension/v1/EntityPlacer.java
+++ /dev/null
@@ -1,50 +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.api.dimension.v1;
-
-import net.minecraft.block.pattern.BlockPattern;
-import net.minecraft.entity.Entity;
-import net.minecraft.server.world.ServerWorld;
-import net.minecraft.util.math.Direction;
-
-/**
- * Responsible for placing an Entity once they have entered a dimension.
- * used in Entity::changeDimension.
- *
- * @deprecated Experimental feature, may be removed or changed without further notice due to potential changes to Dimensions in subsequent versions.
- *
- * @see FabricDimensions
- */
-@Deprecated
-@FunctionalInterface
-public interface EntityPlacer {
-	/**
-	 * Handles the placement of an entity going to a dimension.
-	 * Utilized by {@link FabricDimensions#teleport(Entity, ServerWorld, EntityPlacer)} to specify placement logic when needed.
-	 *
-	 * <p>This method may have side effects such as the creation of a portal in the target dimension,
-	 * or the creation of a chunk loading ticket.
-	 *
-	 * @param portalDir        the direction the portal is facing, meaningless if no portal was used
-	 * @param horizontalOffset the horizontal offset of the entity relative to the front top left corner of the portal, meaningless if no portal was used
-	 * @param verticalOffset   the vertical offset of the entity relative to the front top left corner of the portal, meaningless if no portal was used
-	 * @return a teleportation target, or {@code null} to fall back to further handling
-	 * @apiNote When this method is called, the entity's world is its source dimension.
-	 */
-	/* @Nullable */
-	BlockPattern.TeleportTarget placeEntity(Entity teleported, ServerWorld destination, Direction portalDir, double horizontalOffset, double verticalOffset);
-}
diff --git a/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/api/dimension/v1/FabricDimensions.java b/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/api/dimension/v1/FabricDimensions.java
index ed6e62bf4..eb021f1d9 100644
--- a/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/api/dimension/v1/FabricDimensions.java
+++ b/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/api/dimension/v1/FabricDimensions.java
@@ -17,11 +17,11 @@
 package net.fabricmc.fabric.api.dimension.v1;
 
 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.util.registry.RegistryKey;
-import net.minecraft.world.World;
+import net.minecraft.world.TeleportTarget;
 
 import net.fabricmc.fabric.impl.dimension.FabricDimensionInternals;
 
@@ -37,74 +37,29 @@ public final class FabricDimensions {
 	}
 
 	/**
-	 * Teleports an entity to a different dimension, using custom placement logic.
+	 * Teleports an entity to a different dimension, placing it at the specified destination.
 	 *
-	 * <p>This method behaves as if:
-	 * <pre>{@code teleported.changeDimension(destination)}</pre>
+	 * <p>Using this method will circumvent Vanilla's portal placement code.
 	 *
-	 * <p>If {@code destination} has a default placer, that placer will be used. If {@code destination} is
-	 * the nether or the overworld, the default logic is the vanilla path.
-	 * For any other dimension, the default placement behaviour is undefined.
-	 * When delegating to a placement logic that uses portals, the entity's {@code lastPortalPosition},
-	 * {@code lastPortalDirectionVector}, and {@code lastPortalDirection} fields should be updated
-	 * before calling this method.
-	 *
-	 * <p>After calling this method, {@code teleported} may be invalidated. Callers should use
-	 * the returned entity for any further manipulation.
+	 * <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
-	 * @return the teleported entity, or a clone of it
-	 * @see #teleport(Entity, ServerWorld)
-	 */
-	public static <E extends Entity> E teleport(E teleported, ServerWorld destination) {
-		return teleport(teleported, destination, null);
-	}
-
-	/**
-	 * Teleports an entity to a different dimension, using custom placement logic.
-	 *
-	 * <p>If {@code customPlacement} is {@code null}, this method behaves as if:
-	 * <pre>{@code teleported.changeDimension(destination)}</pre>
-	 * The {@code customPlacement} may itself return {@code null}, in which case
-	 * the default placement logic for that dimension will be run.
-	 *
-	 * <p>If {@code destination} has a default placer, that placer will be used. If {@code destination} is
-	 * the nether or the overworld, the default logic is the vanilla path.
-	 * For any other dimension, the default placement behaviour is undefined.
-	 * When delegating to a placement logic that uses portals, the entity's {@code lastPortalPosition},
-	 * {@code lastPortalDirectionVector}, and {@code lastPortalDirection} fields should be updated
-	 * before calling this method.
-	 *
-	 * <p>After calling this method, {@code teleported} may be invalidated. Callers should use
-	 * the returned entity for any further manipulation.
-	 *
-	 * @param teleported   the entity to teleport
-	 * @param destination  the dimension the entity will be teleported to
-	 * @param customPlacer custom placement logic that will run before the default one,
-	 *                     or {@code null} to use the dimension's default behavior.
-	 * @param <E>          the type of the teleported entity
-	 * @return the teleported entity, or a clone of it
+	 * @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.
 	 * @throws IllegalStateException if this method is called on a client entity
 	 * @apiNote this method must be called from the main server thread
 	 */
-	public static <E extends Entity> E teleport(E teleported, ServerWorld destination, /*Nullable*/ EntityPlacer customPlacer) {
+	@Nullable
+	public static <E extends Entity> E teleport(E teleported, ServerWorld destination, TeleportTarget target) {
+		Preconditions.checkNotNull(target, "A target must be provided");
 		Preconditions.checkState(!teleported.world.isClient, "Entities can only be teleported on the server side");
 
-		return FabricDimensionInternals.changeDimension(teleported, destination, customPlacer);
-	}
-
-	/**
-	 * Register a default placer for a dimension, this is used when an entity is teleported to a dimension without
-	 * a specified {@link EntityPlacer}.
-	 *
-	 * @param registryKey The dimension {@link RegistryKey}
-	 * @param entityPlacer The {@link EntityPlacer}
-	 */
-	public static void registerDefaultPlacer(RegistryKey<World> registryKey, EntityPlacer entityPlacer) {
-		Preconditions.checkState(!FabricDimensionInternals.DEFAULT_PLACERS.containsKey(registryKey), "Only 1 EntityPlacer can be registered per dimension");
-		Preconditions.checkState(!registryKey.getValue().getNamespace().equals("minecraft"), "Minecraft dimensions cannot have a default placer");
-
-		FabricDimensionInternals.DEFAULT_PLACERS.put(registryKey, entityPlacer);
+		return FabricDimensionInternals.changeDimension(teleported, destination, target);
 	}
 }
diff --git a/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/impl/dimension/FabricDimensionInternals.java b/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/impl/dimension/FabricDimensionInternals.java
index 4158530be..df1b9b3e6 100644
--- a/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/impl/dimension/FabricDimensionInternals.java
+++ b/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/impl/dimension/FabricDimensionInternals.java
@@ -16,116 +16,41 @@
 
 package net.fabricmc.fabric.impl.dimension;
 
-import java.util.HashMap;
-import java.util.Map;
-
 import com.google.common.base.Preconditions;
 
-import net.minecraft.block.pattern.BlockPattern;
 import net.minecraft.entity.Entity;
 import net.minecraft.server.world.ServerWorld;
-import net.minecraft.util.math.Direction;
-import net.minecraft.util.registry.RegistryKey;
-import net.minecraft.world.World;
+import net.minecraft.world.TeleportTarget;
 
-import net.fabricmc.fabric.api.dimension.v1.EntityPlacer;
 import net.fabricmc.fabric.api.dimension.v1.FabricDimensions;
-import net.fabricmc.fabric.mixin.dimension.EntityHooks;
 
 public final class FabricDimensionInternals {
+	/**
+	 * The target passed to the last call to {@link FabricDimensions#teleport(Entity, ServerWorld, TeleportTarget)}.
+	 */
+	private static TeleportTarget currentTarget;
+
 	private FabricDimensionInternals() {
 		throw new AssertionError();
 	}
 
-	public static final Map<RegistryKey<World>, EntityPlacer> DEFAULT_PLACERS = new HashMap<>();
-
 	/**
-	 * The entity currently being transported to another dimension.
+	 * Returns the last target set when a user of the API requested teleportation, or null.
 	 */
-	private static final ThreadLocal<Entity> PORTAL_ENTITY = new ThreadLocal<>();
-	/**
-	 * The custom placement logic passed from {@link FabricDimensions#teleport(Entity, ServerWorld, EntityPlacer)}.
-	 */
-	private static EntityPlacer customPlacement;
-
-	/*
-	 * The dimension change hooks consist of two steps:
-	 * - First, we memorize the currently teleported entity, and set required fields
-	 * - Then, we retrieve the teleported entity in the placement logic in PortalForcer#getPortal
-	 *   and use it to call the entity placers
-	 * This lets us use the exact same logic for any entity and prevent the vanilla getPortal (which has unwanted
-	 * side effects) from running, while keeping the patches minimally invasive.
-	 *
-	 * Shortcomings: bugs may arise if another patch cancels the teleportation method between
-	 * #prepareDimensionalTeleportation and #tryFindPlacement, AND a mod calls PortalForcer#getPortal directly
-	 * right after.
-	 */
-
-	public static void prepareDimensionalTeleportation(Entity entity) {
-		Preconditions.checkNotNull(entity);
-		PORTAL_ENTITY.set(entity);
-
-		// Set values used by `PortalForcer#changeDimension` to prevent a NPE crash.
-		EntityHooks access = ((EntityHooks) entity);
-
-		if (entity.getLastNetherPortalDirectionVector() == null) {
-			access.setLastNetherPortalDirectionVector(entity.getRotationVector());
-		}
-
-		if (entity.getLastNetherPortalDirection() == null) {
-			access.setLastNetherPortalDirection(entity.getHorizontalFacing());
-		}
-	}
-
-	/* Nullable */
-	public static BlockPattern.TeleportTarget tryFindPlacement(ServerWorld destination, Direction portalDir, double portalX, double portalY) {
-		Preconditions.checkNotNull(destination);
-		Entity teleported = PORTAL_ENTITY.get();
-		PORTAL_ENTITY.set(null);
-
-		// If the entity is null, the call does not come from a vanilla context
-		if (teleported == null) {
-			return null;
-		}
-
-		// Custom placement logic, falls back to default dimension placement if no placement or target found
-		EntityPlacer customPlacement = FabricDimensionInternals.customPlacement;
-
-		if (customPlacement != null) {
-			BlockPattern.TeleportTarget customTarget = customPlacement.placeEntity(teleported, destination, portalDir, portalX, portalY);
-
-			if (customTarget != null) {
-				return customTarget;
-			}
-		}
-
-		// Default placement logic, falls back to vanilla if not a fabric dimension
-		RegistryKey<World> registryKey = destination.getRegistryKey();
-
-		if (DEFAULT_PLACERS.containsKey(registryKey)) {
-			BlockPattern.TeleportTarget defaultTarget = DEFAULT_PLACERS.get(registryKey).placeEntity(teleported, destination, portalDir, portalX, portalY);
-
-			if (defaultTarget == null) {
-				throw new IllegalStateException("Mod dimension " + destination.getRegistryKey().getValue().toString() + " returned an invalid teleport target");
-			}
-
-			return defaultTarget;
-		}
-
-		// Vanilla / other implementations logic, undefined behaviour on custom dimensions
-		return null;
+	public static TeleportTarget getCustomTarget() {
+		return currentTarget;
 	}
 
 	@SuppressWarnings("unchecked")
-	public static <E extends Entity> E changeDimension(E teleported, ServerWorld dimension, EntityPlacer placement) {
-		assert !teleported.world.isClient : "Entities can only be teleported on the server side";
-		assert Thread.currentThread() == ((ServerWorld) teleported.world).getServer().getThread() : "Entities must be teleported from the main server thread";
+	public static <E extends Entity> E changeDimension(E teleported, ServerWorld dimension, TeleportTarget target) {
+		Preconditions.checkArgument(!teleported.world.isClient, "Entities can only be teleported on the server side");
+		Preconditions.checkArgument(Thread.currentThread() == ((ServerWorld) teleported.world).getServer().getThread(), "Entities must be teleported from the main server thread");
 
 		try {
-			customPlacement = placement;
-			return (E) teleported.changeDimension(dimension);
+			currentTarget = target;
+			return (E) teleported.moveToWorld(dimension);
 		} finally {
-			customPlacement = null;
+			currentTarget = null;
 		}
 	}
 }
diff --git a/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/EntityHooks.java b/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/EntityHooks.java
deleted file mode 100644
index d54040c9b..000000000
--- a/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/EntityHooks.java
+++ /dev/null
@@ -1,33 +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.spongepowered.asm.mixin.Mixin;
-import org.spongepowered.asm.mixin.gen.Accessor;
-
-import net.minecraft.entity.Entity;
-import net.minecraft.util.math.Direction;
-import net.minecraft.util.math.Vec3d;
-
-@Mixin(Entity.class)
-public interface EntityHooks {
-	@Accessor
-	void setLastNetherPortalDirectionVector(Vec3d vec);
-
-	@Accessor
-	void setLastNetherPortalDirection(Direction dir);
-}
diff --git a/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/MixinEntity.java b/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/EntityMixin.java
similarity index 58%
rename from fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/MixinEntity.java
rename to fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/EntityMixin.java
index d250563ca..df760b7a5 100644
--- a/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/MixinEntity.java
+++ b/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/EntityMixin.java
@@ -23,14 +23,25 @@ import org.spongepowered.asm.mixin.injection.callback.CallbackInfoReturnable;
 
 import net.minecraft.entity.Entity;
 import net.minecraft.server.world.ServerWorld;
+import net.minecraft.world.TeleportTarget;
 
 import net.fabricmc.fabric.impl.dimension.FabricDimensionInternals;
 
+/**
+ * This mixin implements {@link Entity#getTeleportTarget(ServerWorld)} for modded dimensions, as Vanilla will
+ * not return a teleport target for anything but Vanilla dimensions.
+ */
 @Mixin(Entity.class)
-public abstract class MixinEntity {
-	// Inject right before the direction vector is retrieved by the game
-	@Inject(method = "changeDimension", at = @At(value = "INVOKE", target = "Lnet/minecraft/entity/Entity;getLastNetherPortalDirectionVector()Lnet/minecraft/util/math/Vec3d;"))
-	private void onGetPortal(ServerWorld targetWorld, CallbackInfoReturnable<Entity> cir) {
-		FabricDimensionInternals.prepareDimensionalTeleportation((Entity) (Object) this);
+public class EntityMixin {
+	@SuppressWarnings("ConstantConditions")
+	@Inject(method = "getTeleportTarget", at = @At("HEAD"), cancellable = true, allow = 1)
+	public void getTeleportTarget(ServerWorld destination, CallbackInfoReturnable<TeleportTarget> cri) {
+		Entity self = (Entity) (Object) this;
+		// Check if a destination has been set for the entity currently being teleported
+		TeleportTarget customTarget = FabricDimensionInternals.getCustomTarget();
+
+		if (customTarget != null) {
+			cri.setReturnValue(customTarget);
+		}
 	}
 }
diff --git a/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/MixinPortalForcer.java b/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/MixinPortalForcer.java
deleted file mode 100644
index 3ce940d34..000000000
--- a/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/MixinPortalForcer.java
+++ /dev/null
@@ -1,55 +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.spongepowered.asm.mixin.Final;
-import org.spongepowered.asm.mixin.Mixin;
-import org.spongepowered.asm.mixin.Shadow;
-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.block.pattern.BlockPattern;
-import net.minecraft.entity.Entity;
-import net.minecraft.server.world.ServerWorld;
-import net.minecraft.util.math.BlockPos;
-import net.minecraft.util.math.Direction;
-import net.minecraft.util.math.Vec3d;
-import net.minecraft.world.PortalForcer;
-
-import net.fabricmc.fabric.impl.dimension.FabricDimensionInternals;
-
-@Mixin(PortalForcer.class)
-public abstract class MixinPortalForcer {
-	@Shadow
-	@Final
-	private ServerWorld world;
-
-	@Inject(method = "usePortal", at = @At(value = "INVOKE", target = "Lnet/minecraft/entity/Entity;getLastNetherPortalDirectionVector()Lnet/minecraft/util/math/Vec3d;"))
-	private void onUsePortal(Entity teleported, float yaw, CallbackInfoReturnable<Boolean> cir) {
-		FabricDimensionInternals.prepareDimensionalTeleportation(teleported);
-	}
-
-	@Inject(method = "getPortal", at = @At("HEAD"), cancellable = true)
-	private void findEntityPlacement(BlockPos pos, Vec3d velocity, Direction portalDir, double portalX, double portalY, boolean player, CallbackInfoReturnable<BlockPattern.TeleportTarget> cir) {
-		BlockPattern.TeleportTarget ret = FabricDimensionInternals.tryFindPlacement(this.world, portalDir, portalX, portalY);
-
-		if (ret != null) {
-			cir.setReturnValue(ret);
-		}
-	}
-}
diff --git a/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/MixinWorld.java b/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/MixinWorld.java
deleted file mode 100644
index 2d73d5ae5..000000000
--- a/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/MixinWorld.java
+++ /dev/null
@@ -1,58 +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.spongepowered.asm.mixin.Mixin;
-import org.spongepowered.asm.mixin.Shadow;
-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.world.World;
-import net.minecraft.world.dimension.DimensionType;
-
-@Mixin(World.class)
-public abstract class MixinWorld {
-	@Shadow
-	public abstract DimensionType getDimension();
-
-	@Shadow
-	private int ambientDarkness;
-
-	/* World.isDay() and World.isNight() enable the day-night cycle as well as some entity behavior
-	 * (such as bees). In vanilla, these methods are hardcoded to only work in the overworld. This
-	 * redirector pretends that all dimensions with a visible sky are DimensionType.OVERWORLD, which
-	 * makes the time checks for modded dimensions work.
-	 *
-	 * Dimension.hasVisibleSky() is true for the overworld, false for the nether and the end, and
-	 * customizable for modded dimensions. It is already used for time checking in other places
-	 * such as clocks.
-	 */
-	@Inject(method = "isDay", at = @At("HEAD"), cancellable = true)
-	private void isDay(CallbackInfoReturnable<Boolean> infoReturnable) {
-		if (getDimension().hasSkyLight()) {
-			infoReturnable.setReturnValue(ambientDarkness < 4);
-		}
-	}
-
-	@Inject(method = "isNight", at = @At("HEAD"), cancellable = true)
-	private void isNight(CallbackInfoReturnable<Boolean> infoReturnable) {
-		if (getDimension().hasSkyLight()) {
-			infoReturnable.setReturnValue(!(ambientDarkness < 4));
-		}
-	}
-}
diff --git a/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/ServerBugfixMixin.java b/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/ServerBugfixMixin.java
new file mode 100644
index 000000000..dc7d60b3f
--- /dev/null
+++ b/fabric-dimensions-v1/src/main/java/net/fabricmc/fabric/mixin/dimension/ServerBugfixMixin.java
@@ -0,0 +1,92 @@
+/*
+ * 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 com.mojang.serialization.Lifecycle;
+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.ModifyVariable;
+import org.spongepowered.asm.mixin.injection.Redirect;
+
+import net.minecraft.nbt.Tag;
+import net.minecraft.resource.DataPackSettings;
+import net.minecraft.server.Main;
+import net.minecraft.util.dynamic.RegistryOps;
+import net.minecraft.util.registry.DynamicRegistryManager;
+import net.minecraft.world.gen.GeneratorOptions;
+import net.minecraft.world.level.LevelInfo;
+import net.minecraft.world.level.LevelProperties;
+import net.minecraft.world.level.storage.LevelStorage;
+
+/**
+ * This Mixin aims to solve a Minecraft Vanilla bug where datapacks are ignored during creation of the
+ * initial LevelProperties when a dedicated server creates a completely new level.
+ *
+ * <p>This also includes the datapacks of loaded Fabric mods, and results in modded dimensions only
+ * being available after restarting the server, once the world has been created.
+ *
+ * <p>This Mixin aims to solve this problem by saving and loading the level.dat file once, after
+ * a new set of level properties is created. This will apply the same logic as reloading the
+ * level.dat after a restart, now including all datapack dimensions.
+ *
+ * <p>See https://bugs.mojang.com/browse/MC-195468 for a related bug report.
+ *
+ * <p>In 1.17: Retest if this bug still occurs without this Mixin by launching a dedicated server with the
+ * dimension testmod, and no world directory. If the dimension is available (i.e. in /execute in, or via
+ * the testmod's commands), then the bug is fixed and this Mixin can be removed.
+ */
+@Mixin(value = Main.class, remap = false)
+public class ServerBugfixMixin {
+	@Unique
+	private static LevelStorage.Session fabric_session;
+
+	@Unique
+	private static DynamicRegistryManager.Impl fabric_dynamicRegistry;
+
+	@Unique
+	private static RegistryOps<Tag> fabric_registryOps;
+
+	@ModifyVariable(at = @At(value = "INVOKE_ASSIGN", target = "Lnet/minecraft/util/registry/DynamicRegistryManager;create()Lnet/minecraft/util/registry/DynamicRegistryManager$Impl;"), method = "main", remap = false, allow = 1)
+	private static DynamicRegistryManager.Impl captureDynamicRegistry(DynamicRegistryManager.Impl value) {
+		fabric_dynamicRegistry = value;
+		return value;
+	}
+
+	@ModifyVariable(at = @At(value = "INVOKE_ASSIGN", target = "Lnet/minecraft/world/level/storage/LevelStorage;createSession(Ljava/lang/String;)Lnet/minecraft/world/level/storage/LevelStorage$Session;"), method = "main", remap = false, allow = 1)
+	private static LevelStorage.Session captureSession(LevelStorage.Session value) {
+		fabric_session = value;
+		return value;
+	}
+
+	@ModifyVariable(at = @At(value = "INVOKE_ASSIGN", target = "Lnet/minecraft/util/dynamic/RegistryOps;of(Lcom/mojang/serialization/DynamicOps;Lnet/minecraft/resource/ResourceManager;Lnet/minecraft/util/registry/DynamicRegistryManager$Impl;)Lnet/minecraft/util/dynamic/RegistryOps;"), method = "main", remap = false, allow = 1)
+	private static RegistryOps<Tag> captureRegistryOps(RegistryOps<Tag> value) {
+		fabric_registryOps = value;
+		return value;
+	}
+
+	@Redirect(method = "main", at = @At(value = "NEW", target = "net/minecraft/world/level/LevelProperties"), remap = false, allow = 1)
+	private static LevelProperties onCreateNewLevelProperties(LevelInfo levelInfo, GeneratorOptions generatorOptions, Lifecycle lifecycle) {
+		DataPackSettings dataPackSettings = levelInfo.getDataPackSettings();
+
+		// Save the level.dat file
+		fabric_session.backupLevelDataFile(fabric_dynamicRegistry, new LevelProperties(levelInfo, generatorOptions, lifecycle));
+
+		// And reload it again, and replace the actual level properties with it
+		return (LevelProperties) fabric_session.readLevelProperties(fabric_registryOps, dataPackSettings);
+	}
+}
diff --git a/fabric-dimensions-v1/src/main/resources/fabric-dimensions-v1.mixins.json b/fabric-dimensions-v1/src/main/resources/fabric-dimensions-v1.mixins.json
index 69e079d2a..50894fe74 100644
--- a/fabric-dimensions-v1/src/main/resources/fabric-dimensions-v1.mixins.json
+++ b/fabric-dimensions-v1/src/main/resources/fabric-dimensions-v1.mixins.json
@@ -3,10 +3,8 @@
   "package": "net.fabricmc.fabric.mixin.dimension",
   "compatibilityLevel": "JAVA_8",
   "mixins": [
-    "EntityHooks",
-    "MixinEntity",
-    "MixinPortalForcer",
-    "MixinWorld"
+    "EntityMixin",
+    "ServerBugfixMixin"
   ],
   "injectors": {
     "defaultRequire": 1
diff --git a/fabric-dimensions-v1/src/testmod/java/net/fabricmc/fabric/test/dimension/FabricDimensionTest.java b/fabric-dimensions-v1/src/testmod/java/net/fabricmc/fabric/test/dimension/FabricDimensionTest.java
index dabb4b1cd..1681440c6 100644
--- a/fabric-dimensions-v1/src/testmod/java/net/fabricmc/fabric/test/dimension/FabricDimensionTest.java
+++ b/fabric-dimensions-v1/src/testmod/java/net/fabricmc/fabric/test/dimension/FabricDimensionTest.java
@@ -16,65 +16,113 @@
 
 package net.fabricmc.fabric.test.dimension;
 
+import static net.minecraft.entity.EntityType.COW;
 import static net.minecraft.server.command.CommandManager.literal;
 
 import com.mojang.brigadier.context.CommandContext;
 import com.mojang.brigadier.exceptions.CommandSyntaxException;
 
 import net.minecraft.block.Blocks;
-import net.minecraft.block.pattern.BlockPattern;
+import net.minecraft.command.CommandException;
 import net.minecraft.entity.Entity;
 import net.minecraft.server.command.ServerCommandSource;
 import net.minecraft.server.network.ServerPlayerEntity;
 import net.minecraft.server.world.ServerWorld;
+import net.minecraft.text.LiteralText;
+import net.minecraft.util.Identifier;
 import net.minecraft.util.math.BlockPos;
-import net.minecraft.util.math.Direction;
+import net.minecraft.util.math.Vec3d;
 import net.minecraft.util.registry.Registry;
 import net.minecraft.util.registry.RegistryKey;
+import net.minecraft.world.TeleportTarget;
 import net.minecraft.world.World;
-import net.minecraft.util.Identifier;
-import net.minecraft.util.math.Vec3d;
+import net.minecraft.world.dimension.DimensionOptions;
+import net.minecraft.world.dimension.DimensionType;
 
 import net.fabricmc.api.ModInitializer;
 import net.fabricmc.fabric.api.command.v1.CommandRegistrationCallback;
 import net.fabricmc.fabric.api.dimension.v1.FabricDimensions;
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerLifecycleEvents;
 
 public class FabricDimensionTest implements ModInitializer {
-	private static RegistryKey<World> dimensionRegistryKey;
+	// The dimension options refer to the JSON-file in the dimension subfolder of the datapack,
+	// which will always share it's ID with the world that is created from it
+	private static final RegistryKey<DimensionOptions> DIMENSION_KEY = RegistryKey.of(
+			Registry.DIMENSION_OPTIONS,
+			new Identifier("fabric_dimension", "void")
+	);
+
+	private static RegistryKey<World> WORLD_KEY = RegistryKey.of(
+			Registry.DIMENSION,
+			DIMENSION_KEY.getValue()
+	);
+
+	private static final RegistryKey<DimensionType> DIMENSION_TYPE_KEY = RegistryKey.of(
+			Registry.DIMENSION_TYPE_KEY,
+			new Identifier("fabric_dimension", "void_type")
+	);
 
 	@Override
 	public void onInitialize() {
 		Registry.register(Registry.CHUNK_GENERATOR, new Identifier("fabric_dimension", "void"), VoidChunkGenerator.CODEC);
 
-		dimensionRegistryKey = RegistryKey.of(Registry.DIMENSION, new Identifier("fabric_dimension", "void"));
+		WORLD_KEY = RegistryKey.of(Registry.DIMENSION, new Identifier("fabric_dimension", "void"));
 
-		FabricDimensions.registerDefaultPlacer(dimensionRegistryKey, FabricDimensionTest::placeEntityInVoid);
+		ServerLifecycleEvents.SERVER_STARTED.register(server -> {
+			ServerWorld overworld = server.getWorld(World.OVERWORLD);
+			ServerWorld world = server.getWorld(WORLD_KEY);
+
+			if (world == null) throw new AssertionError("Test world doesn't exist.");
+
+			Entity entity = COW.create(overworld);
+
+			if (!entity.world.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);
+
+			Entity teleported = FabricDimensions.teleport(entity, world, target);
+
+			if (teleported == null) throw new AssertionError("Entity didn't teleport");
+
+			if (!teleported.world.getRegistryKey().equals(WORLD_KEY)) throw new AssertionError("Target world not reached.");
+
+			if (!teleported.getPos().equals(target.position)) throw new AssertionError("Target Position not reached.");
+		});
 
 		CommandRegistrationCallback.EVENT.register((dispatcher, dedicated) ->
-				dispatcher.register(literal("fabric_dimension_test").executes(FabricDimensionTest.this::executeTestCommand))
+				dispatcher.register(literal("fabric_dimension_test").executes(FabricDimensionTest.this::swapTargeted))
 		);
 	}
 
-	private int executeTestCommand(CommandContext<ServerCommandSource> context) throws CommandSyntaxException {
-		ServerPlayerEntity serverPlayerEntity = context.getSource().getPlayer();
-		ServerWorld serverWorld = serverPlayerEntity.getServerWorld();
+	private int swapTargeted(CommandContext<ServerCommandSource> context) throws CommandSyntaxException {
+		ServerPlayerEntity player = context.getSource().getPlayer();
+		ServerWorld serverWorld = player.getServerWorld();
+		ServerWorld modWorld = getModWorld(context);
 
-		if (!serverWorld.getRegistryKey().equals(dimensionRegistryKey)) {
-			serverPlayerEntity.changeDimension(context.getSource().getMinecraftServer().getWorld(dimensionRegistryKey));
+		if (serverWorld != modWorld) {
+			TeleportTarget target = new TeleportTarget(new Vec3d(0.5, 101, 0.5), Vec3d.ZERO, 0, 0);
+			FabricDimensions.teleport(player, modWorld, target);
+
+			if (player.world != modWorld) {
+				throw new CommandException(new LiteralText("Teleportation failed!"));
+			}
+
+			modWorld.setBlockState(new BlockPos(0, 100, 0), Blocks.DIAMOND_BLOCK.getDefaultState());
+			modWorld.setBlockState(new BlockPos(0, 101, 0), Blocks.TORCH.getDefaultState());
 		} else {
-			FabricDimensions.teleport(serverPlayerEntity, context.getSource().getMinecraftServer().getWorld(World.OVERWORLD), FabricDimensionTest::placeEntity);
+			TeleportTarget target = new TeleportTarget(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);
 		}
 
 		return 1;
 	}
 
-	private static BlockPattern.TeleportTarget placeEntity(Entity teleported, ServerWorld destination, Direction portalDir, double horizontalOffset, double verticalOffset) {
-		return new BlockPattern.TeleportTarget(new Vec3d(0, 100, 0), Vec3d.ZERO, 0);
+	private ServerWorld getModWorld(CommandContext<ServerCommandSource> context) {
+		return getWorld(context, WORLD_KEY);
 	}
 
-	private static BlockPattern.TeleportTarget placeEntityInVoid(Entity teleported, ServerWorld destination, Direction portalDir, double horizontalOffset, double verticalOffset) {
-		destination.setBlockState(new BlockPos(0, 100, 0), Blocks.DIAMOND_BLOCK.getDefaultState());
-		destination.setBlockState(new BlockPos(0, 101, 0), Blocks.TORCH.getDefaultState());
-		return new BlockPattern.TeleportTarget(new Vec3d(0.5, 101, 0.5), Vec3d.ZERO, 0);
+	private ServerWorld getWorld(CommandContext<ServerCommandSource> context, RegistryKey<World> dimensionRegistryKey) {
+		return context.getSource().getMinecraftServer().getWorld(dimensionRegistryKey);
 	}
 }
diff --git a/fabric-dimensions-v1/src/testmod/java/net/fabricmc/fabric/test/dimension/VoidChunkGenerator.java b/fabric-dimensions-v1/src/testmod/java/net/fabricmc/fabric/test/dimension/VoidChunkGenerator.java
index 2b065acec..038819aa2 100644
--- a/fabric-dimensions-v1/src/testmod/java/net/fabricmc/fabric/test/dimension/VoidChunkGenerator.java
+++ b/fabric-dimensions-v1/src/testmod/java/net/fabricmc/fabric/test/dimension/VoidChunkGenerator.java
@@ -37,7 +37,7 @@ public class VoidChunkGenerator extends ChunkGenerator {
 
 	public static final Codec<VoidChunkGenerator> CODEC = RecordCodecBuilder.create((instance) ->
 			instance.group(
-					BiomeSource.field_24713.fieldOf("biome_source")
+					BiomeSource.CODEC.fieldOf("biome_source")
 							.forGetter((generator) -> generator.biomeSource),
 					Codec.BOOL.fieldOf("custom_bool")
 							.forGetter((generator) -> generator.customBool)
@@ -51,7 +51,7 @@ public class VoidChunkGenerator extends ChunkGenerator {
 	}
 
 	@Override
-	protected Codec<? extends ChunkGenerator> method_28506() {
+	protected Codec<? extends ChunkGenerator> getCodec() {
 		return CODEC;
 	}
 
diff --git a/fabric-dimensions-v1/src/testmod/resources/data/minecraft/dimension/fabric_dimension/void.json b/fabric-dimensions-v1/src/testmod/resources/data/fabric_dimension/dimension/void.json
similarity index 100%
rename from fabric-dimensions-v1/src/testmod/resources/data/minecraft/dimension/fabric_dimension/void.json
rename to fabric-dimensions-v1/src/testmod/resources/data/fabric_dimension/dimension/void.json
diff --git a/fabric-dimensions-v1/src/testmod/resources/data/minecraft/dimension_type/fabric_dimension/void_type.json b/fabric-dimensions-v1/src/testmod/resources/data/fabric_dimension/dimension_type/void_type.json
similarity index 92%
rename from fabric-dimensions-v1/src/testmod/resources/data/minecraft/dimension_type/fabric_dimension/void_type.json
rename to fabric-dimensions-v1/src/testmod/resources/data/fabric_dimension/dimension_type/void_type.json
index ba9ac198e..745193cf9 100644
--- a/fabric-dimensions-v1/src/testmod/resources/data/minecraft/dimension_type/fabric_dimension/void_type.json
+++ b/fabric-dimensions-v1/src/testmod/resources/data/fabric_dimension/dimension_type/void_type.json
@@ -1,7 +1,7 @@
 {
   "ultrawarm": false,
   "natural": false,
-  "shrunk": false,
+  "coordinate_scale": 1,
   "ambient_light": 0.1,
   "has_skylight": true,
   "has_ceiling": false,
diff --git a/settings.gradle b/settings.gradle
index e1ba7902b..52d4f22e0 100644
--- a/settings.gradle
+++ b/settings.gradle
@@ -21,7 +21,7 @@ include 'fabric-command-api-v1'
 include 'fabric-containers-v0'
 include 'fabric-content-registries-v0'
 include 'fabric-crash-report-info-v1'
-//include 'fabric-dimensions-v1'
+include 'fabric-dimensions-v1'
 include 'fabric-events-interaction-v0'
 include 'fabric-events-lifecycle-v0'
 include 'fabric-game-rule-api-v1'