diff --git a/fabric-events-lifecycle-v0/build.gradle b/fabric-events-lifecycle-v0/build.gradle
index ca8581d05..3d3226ccb 100644
--- a/fabric-events-lifecycle-v0/build.gradle
+++ b/fabric-events-lifecycle-v0/build.gradle
@@ -1,6 +1,8 @@
 archivesBaseName = "fabric-events-lifecycle-v0"
-version = getSubprojectVersion(project, "0.1.3")
+version = getSubprojectVersion(project, "0.2.0")
 
 dependencies {
 	compile project(path: ':fabric-api-base', configuration: 'dev')
+	compile project(path: ':fabric-item-api-v1', configuration: 'dev')
+	compile project(path: ':fabric-lifecycle-events-v1', configuration: 'dev')
 }
diff --git a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/client/ClientTickCallback.java b/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/client/ClientTickCallback.java
index 653ed4f33..c150a40b2 100644
--- a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/client/ClientTickCallback.java
+++ b/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/client/ClientTickCallback.java
@@ -18,10 +18,16 @@ package net.fabricmc.fabric.api.event.client;
 
 import net.minecraft.client.MinecraftClient;
 
+import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientTickEvents;
 import net.fabricmc.fabric.api.event.Event;
 import net.fabricmc.fabric.api.event.EventFactory;
 
+@Deprecated
 public interface ClientTickCallback {
+	/**
+	 * @deprecated Please use {@link ClientTickEvents#END_CLIENT_TICK}.
+	 */
+	@Deprecated
 	Event<ClientTickCallback> EVENT = EventFactory.createArrayBacked(ClientTickCallback.class,
 			(listeners) -> {
 				if (EventFactory.isProfilingEnabled()) {
diff --git a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/client/ItemTooltipCallback.java b/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/client/ItemTooltipCallback.java
index 7aab92055..f01252953 100644
--- a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/client/ItemTooltipCallback.java
+++ b/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/client/ItemTooltipCallback.java
@@ -25,10 +25,15 @@ import net.minecraft.text.Text;
 import net.fabricmc.fabric.api.event.Event;
 import net.fabricmc.fabric.api.event.EventFactory;
 
+/**
+ * @deprecated Please use {@link net.fabricmc.fabric.api.client.item.v1.ItemTooltipCallback}
+ */
+@Deprecated
 public interface ItemTooltipCallback {
 	/**
 	 * Fired after the game has appended all base tooltip lines to the list.
 	 */
+	@Deprecated
 	Event<ItemTooltipCallback> EVENT = EventFactory.createArrayBacked(ItemTooltipCallback.class, (listeners) -> (stack, tooltipContext, lines) -> {
 		for (ItemTooltipCallback callback : listeners) {
 			callback.getTooltip(stack, tooltipContext, lines);
diff --git a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/server/ServerStartCallback.java b/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/server/ServerStartCallback.java
index e314f9366..45f402f25 100644
--- a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/server/ServerStartCallback.java
+++ b/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/server/ServerStartCallback.java
@@ -21,7 +21,12 @@ import net.minecraft.server.MinecraftServer;
 import net.fabricmc.fabric.api.event.Event;
 import net.fabricmc.fabric.api.event.EventFactory;
 
+@Deprecated
 public interface ServerStartCallback {
+	/**
+	 * @deprecated Please use {@link net.fabricmc.fabric.api.event.lifecycle.v1.ServerLifecycleEvents#SERVER_STARTED}
+	 */
+	@Deprecated
 	Event<ServerStartCallback> EVENT = EventFactory.createArrayBacked(ServerStartCallback.class,
 			(listeners) -> (server) -> {
 				for (ServerStartCallback event : listeners) {
diff --git a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/server/ServerStopCallback.java b/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/server/ServerStopCallback.java
index d44b522a2..ff47a1a03 100644
--- a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/server/ServerStopCallback.java
+++ b/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/server/ServerStopCallback.java
@@ -21,7 +21,12 @@ import net.minecraft.server.MinecraftServer;
 import net.fabricmc.fabric.api.event.Event;
 import net.fabricmc.fabric.api.event.EventFactory;
 
+@Deprecated
 public interface ServerStopCallback {
+	/**
+	 * @deprecated Please use {@link net.fabricmc.fabric.api.event.lifecycle.v1.ServerLifecycleEvents#SERVER_STOPPING}
+	 */
+	@Deprecated
 	Event<ServerStopCallback> EVENT = EventFactory.createArrayBacked(ServerStopCallback.class,
 			(listeners) -> (server) -> {
 				for (ServerStopCallback event : listeners) {
diff --git a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/server/ServerTickCallback.java b/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/server/ServerTickCallback.java
index 56a2bc7fb..9c03dc782 100644
--- a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/server/ServerTickCallback.java
+++ b/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/server/ServerTickCallback.java
@@ -20,8 +20,14 @@ import net.minecraft.server.MinecraftServer;
 
 import net.fabricmc.fabric.api.event.Event;
 import net.fabricmc.fabric.api.event.EventFactory;
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerTickEvents;
 
+@Deprecated
 public interface ServerTickCallback {
+	/**
+	 * @deprecated Please use {@link ServerTickEvents#END_SERVER_TICK}
+	 */
+	@Deprecated
 	Event<ServerTickCallback> EVENT = EventFactory.createArrayBacked(ServerTickCallback.class,
 			(listeners) -> {
 				if (EventFactory.isProfilingEnabled()) {
diff --git a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/world/WorldTickCallback.java b/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/world/WorldTickCallback.java
index 38cda986f..1ee44391a 100644
--- a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/world/WorldTickCallback.java
+++ b/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/api/event/world/WorldTickCallback.java
@@ -18,10 +18,18 @@ package net.fabricmc.fabric.api.event.world;
 
 import net.minecraft.world.World;
 
+import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientTickEvents;
 import net.fabricmc.fabric.api.event.Event;
 import net.fabricmc.fabric.api.event.EventFactory;
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerTickEvents;
 
+@Deprecated
 public interface WorldTickCallback {
+	/**
+	 * @deprecated The new WorldTickCallback has been split into a client and server callback.
+	 * Please use the {@link ServerTickEvents#END_WORLD_TICK server} or {@link ClientTickEvents#END_WORLD_TICK client} callbacks.
+	 */
+	@Deprecated
 	Event<WorldTickCallback> EVENT = EventFactory.createArrayBacked(WorldTickCallback.class,
 			(listeners) -> {
 				if (EventFactory.isProfilingEnabled()) {
diff --git a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/impl/event/lifecycle/LegacyEventInvokers.java b/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/impl/event/lifecycle/LegacyEventInvokers.java
new file mode 100644
index 000000000..5bb4ce1d8
--- /dev/null
+++ b/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/impl/event/lifecycle/LegacyEventInvokers.java
@@ -0,0 +1,37 @@
+/*
+ * 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.event.lifecycle;
+
+import net.fabricmc.api.ModInitializer;
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerLifecycleEvents;
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerTickEvents;
+import net.fabricmc.fabric.api.event.server.ServerStartCallback;
+import net.fabricmc.fabric.api.event.server.ServerStopCallback;
+import net.fabricmc.fabric.api.event.server.ServerTickCallback;
+import net.fabricmc.fabric.api.event.world.WorldTickCallback;
+
+public class LegacyEventInvokers implements ModInitializer {
+	@Override
+	public void onInitialize() {
+		// Allows deprecated events to still be invoked by the newer implementations
+		ServerLifecycleEvents.SERVER_STARTED.register(server -> ServerStartCallback.EVENT.invoker().onStartServer(server));
+		ServerLifecycleEvents.SERVER_STOPPING.register(server -> ServerStopCallback.EVENT.invoker().onStopServer(server));
+		ServerTickEvents.END_SERVER_TICK.register(server -> ServerTickCallback.EVENT.invoker().tick(server));
+		// Tick old events on ServerWorld
+		ServerTickEvents.END_WORLD_TICK.register(world -> WorldTickCallback.EVENT.invoker().tick(world));
+	}
+}
diff --git a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/impl/event/lifecycle/client/LegacyClientEventInvokers.java b/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/impl/event/lifecycle/client/LegacyClientEventInvokers.java
new file mode 100644
index 000000000..f10c7b298
--- /dev/null
+++ b/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/impl/event/lifecycle/client/LegacyClientEventInvokers.java
@@ -0,0 +1,35 @@
+/*
+ * 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.event.lifecycle.client;
+
+import net.fabricmc.api.ClientModInitializer;
+import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientTickEvents;
+import net.fabricmc.fabric.api.event.client.ClientTickCallback;
+import net.fabricmc.fabric.api.event.client.ItemTooltipCallback;
+import net.fabricmc.fabric.api.event.world.WorldTickCallback;
+
+public class LegacyClientEventInvokers implements ClientModInitializer {
+	@Override
+	public void onInitializeClient() {
+		// Allows deprecated events to still be invoked by the newer implementations
+		ClientTickEvents.END_CLIENT_TICK.register(client -> ClientTickCallback.EVENT.invoker().tick(client));
+		// Tick old events on ClientWorld
+		ClientTickEvents.END_WORLD_TICK.register(world -> WorldTickCallback.EVENT.invoker().tick(world));
+		// This is part of item api now.
+		net.fabricmc.fabric.api.client.item.v1.ItemTooltipCallback.EVENT.register((stack, context, lines) -> ItemTooltipCallback.EVENT.invoker().getTooltip(stack, context, lines));
+	}
+}
diff --git a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/MixinMinecraftClient.java b/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/MixinMinecraftClient.java
deleted file mode 100644
index a34368aea..000000000
--- a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/MixinMinecraftClient.java
+++ /dev/null
@@ -1,34 +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.event.lifecycle;
-
-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.callback.CallbackInfo;
-
-import net.minecraft.client.MinecraftClient;
-
-import net.fabricmc.fabric.api.event.client.ClientTickCallback;
-
-@Mixin(MinecraftClient.class)
-public class MixinMinecraftClient {
-	@Inject(at = @At("RETURN"), method = "tick")
-	public void tick(CallbackInfo info) {
-		ClientTickCallback.EVENT.invoker().tick((MinecraftClient) (Object) this);
-	}
-}
diff --git a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/MixinWorld.java b/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/MixinWorld.java
deleted file mode 100644
index 9b797509b..000000000
--- a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/MixinWorld.java
+++ /dev/null
@@ -1,35 +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.event.lifecycle;
-
-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.callback.CallbackInfo;
-
-import net.minecraft.world.World;
-
-import net.fabricmc.fabric.api.event.world.WorldTickCallback;
-
-@Mixin(World.class)
-public class MixinWorld {
-	// TODO split into ClientWorld/ServerWorld ticks? mmm need more mappings
-	@Inject(at = @At("RETURN"), method = "tickBlockEntities")
-	public void tickBlockEntitiesAfter(CallbackInfo info) {
-		WorldTickCallback.EVENT.invoker().tick((World) (Object) this);
-	}
-}
diff --git a/fabric-events-lifecycle-v0/src/main/resources/fabric-events-lifecycle-v0.mixins.json b/fabric-events-lifecycle-v0/src/main/resources/fabric-events-lifecycle-v0.mixins.json
deleted file mode 100644
index 738bbd9a1..000000000
--- a/fabric-events-lifecycle-v0/src/main/resources/fabric-events-lifecycle-v0.mixins.json
+++ /dev/null
@@ -1,16 +0,0 @@
-{
-  "required": true,
-  "package": "net.fabricmc.fabric.mixin.event.lifecycle",
-  "compatibilityLevel": "JAVA_8",
-  "mixins": [
-    "MixinMinecraftServer",
-    "MixinWorld"
-  ],
-  "client": [
-    "MixinItemStack",
-    "MixinMinecraftClient"
-  ],
-  "injectors": {
-    "defaultRequire": 1
-  }
-}
diff --git a/fabric-events-lifecycle-v0/src/main/resources/fabric.mod.json b/fabric-events-lifecycle-v0/src/main/resources/fabric.mod.json
index c6639d07f..7eb6a935f 100644
--- a/fabric-events-lifecycle-v0/src/main/resources/fabric.mod.json
+++ b/fabric-events-lifecycle-v0/src/main/resources/fabric.mod.json
@@ -15,12 +15,19 @@
   "authors": [
     "FabricMC"
   ],
+  "entrypoints": {
+    "main": [
+      "net.fabricmc.fabric.impl.event.lifecycle.LegacyEventInvokers"
+    ],
+    "client": [
+      "net.fabricmc.fabric.impl.event.lifecycle.client.LegacyClientEventInvokers"
+    ]
+  },
   "depends": {
     "fabricloader": ">=0.4.0",
-    "fabric-api-base": "*"
+    "fabric-api-base": "*",
+    "fabric-item-api-v1": "*",
+    "fabric-lifecycle-events-v1": "*"
   },
-  "description": "Events for the game's lifecycle.",
-  "mixins": [
-    "fabric-events-lifecycle-v0.mixins.json"
-  ]
+  "description": "Legacy events for the game's lifecycle, superseded by fabric-lifecycle-events-v1 and fabric-item-api-v1."
 }
diff --git a/fabric-events-lifecycle-v0/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/legacy/LegacyLifecycleEventsTest.java b/fabric-events-lifecycle-v0/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/legacy/LegacyLifecycleEventsTest.java
new file mode 100644
index 000000000..7d3091d56
--- /dev/null
+++ b/fabric-events-lifecycle-v0/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/legacy/LegacyLifecycleEventsTest.java
@@ -0,0 +1,64 @@
+/*
+ * 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.test.event.lifecycle.legacy;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import org.apache.logging.log4j.LogManager;
+import org.apache.logging.log4j.Logger;
+
+import net.minecraft.util.registry.RegistryKey;
+import net.minecraft.world.World;
+
+import net.fabricmc.api.ModInitializer;
+import net.fabricmc.fabric.api.event.server.ServerStartCallback;
+import net.fabricmc.fabric.api.event.server.ServerStopCallback;
+import net.fabricmc.fabric.api.event.server.ServerTickCallback;
+import net.fabricmc.fabric.api.event.world.WorldTickCallback;
+
+public class LegacyLifecycleEventsTest implements ModInitializer {
+	public static final Logger LOGGER = LogManager.getLogger("LegacyLifecycleEventsTest");
+	private Map<RegistryKey<World>, Integer> tickTracker = new HashMap<>();
+
+	@Override
+	public void onInitialize() {
+		ServerTickCallback.EVENT.register(server -> {
+			if (server.getTicks() % 200 == 0) { // Log every 200 ticks to verify the tick callback works on the server
+				LOGGER.info("Ticked Server at " + server.getTicks() + " ticks. (Legacy)");
+			}
+		});
+
+		ServerStartCallback.EVENT.register(server -> {
+			LOGGER.info("Started Server! (Legacy)");
+		});
+
+		ServerStopCallback.EVENT.register(server -> {
+			LOGGER.info("Stopping Server! (Legacy)");
+		});
+
+		WorldTickCallback.EVENT.register(world -> {
+			final int worldTicks = tickTracker.computeIfAbsent(world.getRegistryKey(), k -> 0);
+
+			if (worldTicks % 200 == 0) { // Log every 200 ticks to verify the tick callback works on the server world
+				LOGGER.info("[LEGACY] Ticked World " + world.getRegistryKey().getValue() + " - " + worldTicks + " ticks: " + world.getClass().getName());
+			}
+
+			this.tickTracker.put(world.getRegistryKey(), worldTicks + 1);
+		});
+	}
+}
diff --git a/fabric-events-lifecycle-v0/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/legacy/client/LegacyClientLifecycleEventsTest.java b/fabric-events-lifecycle-v0/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/legacy/client/LegacyClientLifecycleEventsTest.java
new file mode 100644
index 000000000..810fb9f80
--- /dev/null
+++ b/fabric-events-lifecycle-v0/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/legacy/client/LegacyClientLifecycleEventsTest.java
@@ -0,0 +1,43 @@
+/*
+ * 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.test.event.lifecycle.legacy.client;
+
+import net.minecraft.text.LiteralText;
+
+import net.fabricmc.api.ClientModInitializer;
+import net.fabricmc.fabric.api.event.client.ClientTickCallback;
+import net.fabricmc.fabric.api.event.client.ItemTooltipCallback;
+import net.fabricmc.fabric.test.event.lifecycle.legacy.LegacyLifecycleEventsTest;
+
+public class LegacyClientLifecycleEventsTest implements ClientModInitializer {
+	private int ticks;
+
+	@Override
+	public void onInitializeClient() {
+		ClientTickCallback.EVENT.register(client -> {
+			this.ticks++; // Just track our own tick since the client doesn't have a ticks value.
+
+			if (this.ticks % 200 == 0) {
+				LegacyLifecycleEventsTest.LOGGER.info("Ticked Client at " + this.ticks + " ticks. (Legacy)");
+			}
+		});
+
+		ItemTooltipCallback.EVENT.register((stack, context, lines) -> {
+			lines.add(new LiteralText("A Legacy Tooltip"));
+		});
+	}
+}
diff --git a/fabric-events-lifecycle-v0/src/testmod/resources/fabric.mod.json b/fabric-events-lifecycle-v0/src/testmod/resources/fabric.mod.json
new file mode 100644
index 000000000..c98975051
--- /dev/null
+++ b/fabric-events-lifecycle-v0/src/testmod/resources/fabric.mod.json
@@ -0,0 +1,19 @@
+{
+  "schemaVersion": 1,
+  "id": "fabric-events-lifecycle-v0-testmod",
+  "name": "Fabric Events Lifecycle (v0) Test Mod",
+  "version": "1.0.0",
+  "environment": "*",
+  "license": "Apache-2.0",
+  "depends": {
+    "fabric-events-lifecycle-v0": "*"
+  },
+  "entrypoints": {
+    "main": [
+      "net.fabricmc.fabric.test.event.lifecycle.legacy.LegacyLifecycleEventsTest"
+    ],
+    "client": [
+      "net.fabricmc.fabric.test.event.lifecycle.legacy.client.LegacyClientLifecycleEventsTest"
+    ]
+  }
+}
diff --git a/fabric-item-api-v1/build.gradle b/fabric-item-api-v1/build.gradle
new file mode 100644
index 000000000..8be7121f9
--- /dev/null
+++ b/fabric-item-api-v1/build.gradle
@@ -0,0 +1,6 @@
+archivesBaseName = "fabric-item-api-v1"
+version = getSubprojectVersion(project, "1.0.0")
+
+dependencies {
+	compile project(path: ':fabric-api-base', configuration: 'dev')
+}
diff --git a/fabric-item-api-v1/src/main/java/net/fabricmc/fabric/api/client/item/v1/ItemTooltipCallback.java b/fabric-item-api-v1/src/main/java/net/fabricmc/fabric/api/client/item/v1/ItemTooltipCallback.java
new file mode 100644
index 000000000..60c58eaf1
--- /dev/null
+++ b/fabric-item-api-v1/src/main/java/net/fabricmc/fabric/api/client/item/v1/ItemTooltipCallback.java
@@ -0,0 +1,48 @@
+/*
+ * 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.client.item.v1;
+
+import java.util.List;
+
+import net.minecraft.client.item.TooltipContext;
+import net.minecraft.item.ItemStack;
+import net.minecraft.text.Text;
+
+import net.fabricmc.api.EnvType;
+import net.fabricmc.api.Environment;
+import net.fabricmc.fabric.api.event.Event;
+import net.fabricmc.fabric.api.event.EventFactory;
+
+@Environment(EnvType.CLIENT)
+public interface ItemTooltipCallback {
+	/**
+	 * Fired after the game has appended all base tooltip lines to the list.
+	 */
+	Event<ItemTooltipCallback> EVENT = EventFactory.createArrayBacked(ItemTooltipCallback.class, callbacks -> (stack, context, lines) -> {
+		for (ItemTooltipCallback callback : callbacks) {
+			callback.getTooltip(stack, context, lines);
+		}
+	});
+
+	/**
+	 * Called when an item stack's tooltip is rendered. Text added to {@code lines} will be
+	 * rendered with the tooltip.
+	 *
+	 * @param lines the list containing the lines of text displayed on the stack's tooltip
+	 */
+	void getTooltip(ItemStack stack, TooltipContext context, List<Text> lines);
+}
diff --git a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/MixinItemStack.java b/fabric-item-api-v1/src/main/java/net/fabricmc/fabric/mixin/item/client/ItemStackMixin.java
similarity index 89%
rename from fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/MixinItemStack.java
rename to fabric-item-api-v1/src/main/java/net/fabricmc/fabric/mixin/item/client/ItemStackMixin.java
index c6f025066..60efa8ac9 100644
--- a/fabric-events-lifecycle-v0/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/MixinItemStack.java
+++ b/fabric-item-api-v1/src/main/java/net/fabricmc/fabric/mixin/item/client/ItemStackMixin.java
@@ -14,7 +14,7 @@
  * limitations under the License.
  */
 
-package net.fabricmc.fabric.mixin.event.lifecycle;
+package net.fabricmc.fabric.mixin.item.client;
 
 import java.util.List;
 
@@ -28,10 +28,10 @@ import net.minecraft.entity.player.PlayerEntity;
 import net.minecraft.item.ItemStack;
 import net.minecraft.text.Text;
 
-import net.fabricmc.fabric.api.event.client.ItemTooltipCallback;
+import net.fabricmc.fabric.api.client.item.v1.ItemTooltipCallback;
 
 @Mixin(ItemStack.class)
-public class MixinItemStack {
+public abstract class ItemStackMixin {
 	@Inject(method = "getTooltip", at = @At("RETURN"))
 	private void getTooltip(PlayerEntity entity, TooltipContext tooltipContext, CallbackInfoReturnable<List<Text>> info) {
 		ItemTooltipCallback.EVENT.invoker().getTooltip((ItemStack) (Object) this, tooltipContext, info.getReturnValue());
diff --git a/fabric-item-api-v1/src/main/resources/assets/fabric-item-api-v1/icon.png b/fabric-item-api-v1/src/main/resources/assets/fabric-item-api-v1/icon.png
new file mode 100644
index 000000000..2931efbf6
Binary files /dev/null and b/fabric-item-api-v1/src/main/resources/assets/fabric-item-api-v1/icon.png differ
diff --git a/fabric-item-api-v1/src/main/resources/fabric-item-api-v1.mixins.json b/fabric-item-api-v1/src/main/resources/fabric-item-api-v1.mixins.json
new file mode 100644
index 000000000..c616f9331
--- /dev/null
+++ b/fabric-item-api-v1/src/main/resources/fabric-item-api-v1.mixins.json
@@ -0,0 +1,11 @@
+{
+  "required": true,
+  "package": "net.fabricmc.fabric.mixin.item",
+  "compatibilityLevel": "JAVA_8",
+  "client": [
+    "client.ItemStackMixin"
+  ],
+  "injectors": {
+    "defaultRequire": 1
+  }
+}
diff --git a/fabric-item-api-v1/src/main/resources/fabric.mod.json b/fabric-item-api-v1/src/main/resources/fabric.mod.json
new file mode 100644
index 000000000..111fe5731
--- /dev/null
+++ b/fabric-item-api-v1/src/main/resources/fabric.mod.json
@@ -0,0 +1,26 @@
+{
+  "schemaVersion": 1,
+  "id": "fabric-item-api-v1",
+  "name": "Fabric Item API (v1)",
+  "version": "${version}",
+  "environment": "*",
+  "license": "Apache-2.0",
+  "icon": "assets/fabric-item-api-v1/icon.png",
+  "contact": {
+    "homepage": "https://fabricmc.net",
+    "irc": "irc://irc.esper.net:6667/fabric",
+    "issues": "https://github.com/FabricMC/fabric/issues",
+    "sources": "https://github.com/FabricMC/fabric"
+  },
+  "authors": [
+    "FabricMC"
+  ],
+  "mixins": [
+    "fabric-item-api-v1.mixins.json"
+  ],
+  "depends": {
+    "fabricloader": ">=0.4.0",
+    "fabric-api-base": "*"
+  },
+  "description": "Hooks for items"
+}
diff --git a/fabric-item-api-v1/src/testmod/java/net/fabricmc/fabric/test/item/client/TooltipTests.java b/fabric-item-api-v1/src/testmod/java/net/fabricmc/fabric/test/item/client/TooltipTests.java
new file mode 100644
index 000000000..2c32a5c9e
--- /dev/null
+++ b/fabric-item-api-v1/src/testmod/java/net/fabricmc/fabric/test/item/client/TooltipTests.java
@@ -0,0 +1,36 @@
+/*
+ * 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.test.item.client;
+
+import net.minecraft.text.LiteralText;
+import net.minecraft.util.Formatting;
+
+import net.fabricmc.api.ClientModInitializer;
+import net.fabricmc.api.EnvType;
+import net.fabricmc.api.Environment;
+import net.fabricmc.fabric.api.client.item.v1.ItemTooltipCallback;
+
+@Environment(EnvType.CLIENT)
+public class TooltipTests implements ClientModInitializer {
+	@Override
+	public void onInitializeClient() {
+		// Adds a tooltip to all items so testing can be verified easily.
+		ItemTooltipCallback.EVENT.register((stack, context, lines) -> {
+			lines.add(new LiteralText("Fancy Tooltips").formatted(Formatting.LIGHT_PURPLE));
+		});
+	}
+}
diff --git a/fabric-item-api-v1/src/testmod/resources/fabric.mod.json b/fabric-item-api-v1/src/testmod/resources/fabric.mod.json
new file mode 100644
index 000000000..b566979aa
--- /dev/null
+++ b/fabric-item-api-v1/src/testmod/resources/fabric.mod.json
@@ -0,0 +1,16 @@
+{
+  "schemaVersion": 1,
+  "id": "fabric-item-api-v1-testmod",
+  "name": "Fabric Item API (v1) Test Mod",
+  "version": "1.0.0",
+  "environment": "*",
+  "license": "Apache-2.0",
+  "depends": {
+    "fabric-item-api-v1": "*"
+  },
+  "entrypoints": {
+    "client": [
+      "net.fabricmc.fabric.test.item.client.TooltipTests"
+    ]
+  }
+}
diff --git a/fabric-lifecycle-events-v1/build.gradle b/fabric-lifecycle-events-v1/build.gradle
new file mode 100644
index 000000000..45212c672
--- /dev/null
+++ b/fabric-lifecycle-events-v1/build.gradle
@@ -0,0 +1,6 @@
+archivesBaseName = "fabric-lifecycle-events-v1"
+version = getSubprojectVersion(project, "1.0.0")
+
+dependencies {
+	compile project(path: ':fabric-api-base', configuration: 'dev')
+}
diff --git a/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/client/event/lifecycle/v1/ClientBlockEntityEvents.java b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/client/event/lifecycle/v1/ClientBlockEntityEvents.java
new file mode 100644
index 000000000..ec901429b
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/client/event/lifecycle/v1/ClientBlockEntityEvents.java
@@ -0,0 +1,88 @@
+/*
+ * 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.client.event.lifecycle.v1;
+
+import net.minecraft.block.entity.BlockEntity;
+import net.minecraft.client.world.ClientWorld;
+import net.minecraft.util.profiler.Profiler;
+
+import net.fabricmc.api.EnvType;
+import net.fabricmc.api.Environment;
+import net.fabricmc.fabric.api.event.Event;
+import net.fabricmc.fabric.api.event.EventFactory;
+
+@Environment(EnvType.CLIENT)
+public final class ClientBlockEntityEvents {
+	private ClientBlockEntityEvents() {
+	}
+
+	/**
+	 * Called when a BlockEntity is loaded into a ClientWorld.
+	 *
+	 * <p>When this event is called, the block entity is already in the world.
+	 */
+	public static final Event<ClientBlockEntityEvents.Load> BLOCK_ENTITY_LOAD = EventFactory.createArrayBacked(ClientBlockEntityEvents.Load.class, callbacks -> (blockEntity, world) -> {
+		if (EventFactory.isProfilingEnabled()) {
+			final Profiler profiler = world.getProfiler();
+			profiler.push("fabricClientBlockEntityLoad");
+
+			for (ClientBlockEntityEvents.Load callback : callbacks) {
+				profiler.push(EventFactory.getHandlerName(callback));
+				callback.onLoad(blockEntity, world);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (ClientBlockEntityEvents.Load callback : callbacks) {
+				callback.onLoad(blockEntity, world);
+			}
+		}
+	});
+
+	/**
+	 * Called when a BlockEntity is about to be unloaded from a ClientWorld.
+	 *
+	 * <p>When this event is called, the block entity is still present on the world.
+	 */
+	public static final Event<ClientBlockEntityEvents.Unload> BLOCK_ENTITY_UNLOAD = EventFactory.createArrayBacked(ClientBlockEntityEvents.Unload.class, callbacks -> (blockEntity, world) -> {
+		if (EventFactory.isProfilingEnabled()) {
+			final Profiler profiler = world.getProfiler();
+			profiler.push("fabricClientBlockEntityUnload");
+
+			for (ClientBlockEntityEvents.Unload callback : callbacks) {
+				profiler.push(EventFactory.getHandlerName(callback));
+				callback.onUnload(blockEntity, world);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (ClientBlockEntityEvents.Unload callback : callbacks) {
+				callback.onUnload(blockEntity, world);
+			}
+		}
+	});
+
+	public interface Load {
+		void onLoad(BlockEntity blockEntity, ClientWorld world);
+	}
+
+	public interface Unload {
+		void onUnload(BlockEntity blockEntity, ClientWorld world);
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/client/event/lifecycle/v1/ClientChunkEvents.java b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/client/event/lifecycle/v1/ClientChunkEvents.java
new file mode 100644
index 000000000..7ce0387ad
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/client/event/lifecycle/v1/ClientChunkEvents.java
@@ -0,0 +1,88 @@
+/*
+ * 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.client.event.lifecycle.v1;
+
+import net.minecraft.client.world.ClientWorld;
+import net.minecraft.util.profiler.Profiler;
+import net.minecraft.world.chunk.WorldChunk;
+
+import net.fabricmc.api.EnvType;
+import net.fabricmc.api.Environment;
+import net.fabricmc.fabric.api.event.Event;
+import net.fabricmc.fabric.api.event.EventFactory;
+
+@Environment(EnvType.CLIENT)
+public final class ClientChunkEvents {
+	private ClientChunkEvents() {
+	}
+
+	/**
+	 * Called when a chunk is loaded into a ClientWorld.
+	 *
+	 * <p>When this event is called, the chunk is already in the world.
+	 */
+	public static final Event<ClientChunkEvents.Load> CHUNK_LOAD = EventFactory.createArrayBacked(ClientChunkEvents.Load.class, callbacks -> (clientWorld, chunk) -> {
+		if (EventFactory.isProfilingEnabled()) {
+			Profiler profiler = clientWorld.getProfiler();
+			profiler.push("fabricClientChunkLoad");
+
+			for (ClientChunkEvents.Load callback : callbacks) {
+				profiler.push(EventFactory.getHandlerName(callback));
+				callback.onChunkLoad(clientWorld, chunk);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (ClientChunkEvents.Load callback : callbacks) {
+				callback.onChunkLoad(clientWorld, chunk);
+			}
+		}
+	});
+
+	/**
+	 * Called when a chunk is about to be unloaded from a ClientWorld.
+	 *
+	 * <p>When this event is called, the chunk is still present in the world.
+	 */
+	public static final Event<ClientChunkEvents.Unload> CHUNK_UNLOAD = EventFactory.createArrayBacked(ClientChunkEvents.Unload.class, callbacks -> (clientWorld, chunk) -> {
+		if (EventFactory.isProfilingEnabled()) {
+			final Profiler profiler = clientWorld.getProfiler();
+			profiler.push("fabricClientChunkUnload");
+
+			for (ClientChunkEvents.Unload callback : callbacks) {
+				profiler.push(EventFactory.getHandlerName(callback));
+				callback.onChunkUnload(clientWorld, chunk);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (ClientChunkEvents.Unload callback : callbacks) {
+				callback.onChunkUnload(clientWorld, chunk);
+			}
+		}
+	});
+
+	public interface Load {
+		void onChunkLoad(ClientWorld world, WorldChunk chunk);
+	}
+
+	public interface Unload {
+		void onChunkUnload(ClientWorld world, WorldChunk chunk);
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/client/event/lifecycle/v1/ClientEntityEvents.java b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/client/event/lifecycle/v1/ClientEntityEvents.java
new file mode 100644
index 000000000..c8ced1a4b
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/client/event/lifecycle/v1/ClientEntityEvents.java
@@ -0,0 +1,88 @@
+/*
+ * 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.client.event.lifecycle.v1;
+
+import net.minecraft.client.world.ClientWorld;
+import net.minecraft.entity.Entity;
+import net.minecraft.util.profiler.Profiler;
+
+import net.fabricmc.api.EnvType;
+import net.fabricmc.api.Environment;
+import net.fabricmc.fabric.api.event.Event;
+import net.fabricmc.fabric.api.event.EventFactory;
+
+@Environment(EnvType.CLIENT)
+public final class ClientEntityEvents {
+	public ClientEntityEvents() {
+	}
+
+	/**
+	 * Called when an Entity is loaded into a ClientWorld.
+	 *
+	 * <p>When this event is called, the chunk is already in the world.
+	 */
+	public static final Event<ClientEntityEvents.Load> ENTITY_LOAD = EventFactory.createArrayBacked(ClientEntityEvents.Load.class, callbacks -> (entity, world) -> {
+		if (EventFactory.isProfilingEnabled()) {
+			final Profiler profiler = world.getProfiler();
+			profiler.push("fabricClientEntityLoad");
+
+			for (ClientEntityEvents.Load callback : callbacks) {
+				profiler.push(EventFactory.getHandlerName(callback));
+				callback.onLoad(entity, world);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (ClientEntityEvents.Load callback : callbacks) {
+				callback.onLoad(entity, world);
+			}
+		}
+	});
+
+	/**
+	 * Called when an Entity is about to be unloaded from a ClientWorld.
+	 *
+	 * <p>When this event is called, the entity is still present in the world.
+	 */
+	public static final Event<ClientEntityEvents.Unload> ENTITY_UNLOAD = EventFactory.createArrayBacked(ClientEntityEvents.Unload.class, callbacks -> (entity, world) -> {
+		if (EventFactory.isProfilingEnabled()) {
+			final Profiler profiler = world.getProfiler();
+			profiler.push("fabricClientEntityLoad");
+
+			for (ClientEntityEvents.Unload callback : callbacks) {
+				profiler.push(EventFactory.getHandlerName(callback));
+				callback.onUnload(entity, world);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (ClientEntityEvents.Unload callback : callbacks) {
+				callback.onUnload(entity, world);
+			}
+		}
+	});
+
+	public interface Load {
+		void onLoad(Entity entity, ClientWorld world);
+	}
+
+	public interface Unload {
+		void onUnload(Entity entity, ClientWorld world);
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/client/event/lifecycle/v1/ClientLifecycleEvents.java b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/client/event/lifecycle/v1/ClientLifecycleEvents.java
new file mode 100644
index 000000000..88ce5cac4
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/client/event/lifecycle/v1/ClientLifecycleEvents.java
@@ -0,0 +1,61 @@
+/*
+ * 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.client.event.lifecycle.v1;
+
+import net.minecraft.client.MinecraftClient;
+
+import net.fabricmc.api.EnvType;
+import net.fabricmc.api.Environment;
+import net.fabricmc.fabric.api.event.Event;
+import net.fabricmc.fabric.api.event.EventFactory;
+
+@Environment(EnvType.CLIENT)
+public final class ClientLifecycleEvents {
+	private ClientLifecycleEvents() {
+	}
+
+	/**
+	 * Called when Minecraft has started and it's client about to tick for the first time.
+	 *
+	 * <p>This occurs while the splash screen is displayed.
+	 */
+	public static final Event<ClientStarted> CLIENT_STARTED = EventFactory.createArrayBacked(ClientStarted.class, callbacks -> client -> {
+		for (ClientStarted callback : callbacks) {
+			callback.onClientStarted(client);
+		}
+	});
+
+	/**
+	 * Called when Minecraft's client begins to stop.
+	 * This is caused by quitting while in game, or closing the game window.
+	 *
+	 * <p>This will be called before the integrated server is stopped if it is running.
+	 */
+	public static final Event<ClientStopping> CLIENT_STOPPING = EventFactory.createArrayBacked(ClientStopping.class, callbacks -> client -> {
+		for (ClientStopping callback : callbacks) {
+			callback.onClientStopping(client);
+		}
+	});
+
+	public interface ClientStarted {
+		void onClientStarted(MinecraftClient client);
+	}
+
+	public interface ClientStopping {
+		void onClientStopping(MinecraftClient client);
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/client/event/lifecycle/v1/ClientTickEvents.java b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/client/event/lifecycle/v1/ClientTickEvents.java
new file mode 100644
index 000000000..07b3eb9c1
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/client/event/lifecycle/v1/ClientTickEvents.java
@@ -0,0 +1,138 @@
+/*
+ * 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.client.event.lifecycle.v1;
+
+import net.minecraft.client.MinecraftClient;
+import net.minecraft.client.world.ClientWorld;
+import net.minecraft.util.profiler.Profiler;
+
+import net.fabricmc.api.EnvType;
+import net.fabricmc.api.Environment;
+import net.fabricmc.fabric.api.event.Event;
+import net.fabricmc.fabric.api.event.EventFactory;
+
+@Environment(EnvType.CLIENT)
+public final class ClientTickEvents {
+	public ClientTickEvents() {
+	}
+
+	/**
+	 * Called at the start of the client tick.
+	 */
+	public static final Event<StartTick> START_CLIENT_TICK = EventFactory.createArrayBacked(StartTick.class, callbacks -> client -> {
+		if (EventFactory.isProfilingEnabled()) {
+			final Profiler profiler = client.getProfiler();
+			profiler.push("fabricStartClientTick");
+
+			for (StartTick event : callbacks) {
+				profiler.push(EventFactory.getHandlerName(event));
+				event.onStartTick(client);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (StartTick event : callbacks) {
+				event.onStartTick(client);
+			}
+		}
+	});
+
+	/**
+	 * Called at the end of the client tick.
+	 */
+	public static final Event<EndTick> END_CLIENT_TICK = EventFactory.createArrayBacked(EndTick.class, callbacks -> client -> {
+		if (EventFactory.isProfilingEnabled()) {
+			final Profiler profiler = client.getProfiler();
+			profiler.push("fabricEndClientTick");
+
+			for (EndTick event : callbacks) {
+				profiler.push(EventFactory.getHandlerName(event));
+				event.onEndTick(client);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (EndTick event : callbacks) {
+				event.onEndTick(client);
+			}
+		}
+	});
+
+	/**
+	 * Called at the start of a ClientWorld's tick.
+	 */
+	public static final Event<StartWorldTick> START_WORLD_TICK = EventFactory.createArrayBacked(StartWorldTick.class, callbacks -> world -> {
+		if (EventFactory.isProfilingEnabled()) {
+			final Profiler profiler = world.getProfiler();
+			profiler.push("fabricStartClientWorldTick");
+
+			for (StartWorldTick callback : callbacks) {
+				profiler.push(EventFactory.getHandlerName(callback));
+				callback.onStartTick(world);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (StartWorldTick callback : callbacks) {
+				callback.onStartTick(world);
+			}
+		}
+	});
+
+	/**
+	 * Called at the end of a ClientWorld's tick.
+	 *
+	 * <p>End of world tick may be used to start async computations for the next tick.
+	 */
+	public static final Event<EndWorldTick> END_WORLD_TICK = EventFactory.createArrayBacked(EndWorldTick.class, callbacks -> world -> {
+		if (EventFactory.isProfilingEnabled()) {
+			final Profiler profiler = world.getProfiler();
+			profiler.push("fabricEndClientWorldTick");
+
+			for (EndWorldTick callback : callbacks) {
+				profiler.push(EventFactory.getHandlerName(callback));
+				callback.onEndTick(world);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (EndWorldTick callback : callbacks) {
+				callback.onEndTick(world);
+			}
+		}
+	});
+
+	public interface StartTick {
+		void onStartTick(MinecraftClient client);
+	}
+
+	public interface EndTick {
+		void onEndTick(MinecraftClient client);
+	}
+
+	public interface StartWorldTick {
+		void onStartTick(ClientWorld world);
+	}
+
+	public interface EndWorldTick {
+		void onEndTick(ClientWorld world);
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/event/lifecycle/v1/ServerBlockEntityEvents.java b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/event/lifecycle/v1/ServerBlockEntityEvents.java
new file mode 100644
index 000000000..5418bfb1b
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/event/lifecycle/v1/ServerBlockEntityEvents.java
@@ -0,0 +1,85 @@
+/*
+ * 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.event.lifecycle.v1;
+
+import net.minecraft.block.entity.BlockEntity;
+import net.minecraft.server.world.ServerWorld;
+import net.minecraft.util.profiler.Profiler;
+
+import net.fabricmc.fabric.api.event.Event;
+import net.fabricmc.fabric.api.event.EventFactory;
+
+public final class ServerBlockEntityEvents {
+	private ServerBlockEntityEvents() {
+	}
+
+	/**
+	 * Called when an BlockEntity is loaded into a ServerWorld.
+	 *
+	 * <p>When this is event is called, the block entity is already in the world.
+	 */
+	public static final Event<ServerBlockEntityEvents.Load> BLOCK_ENTITY_LOAD = EventFactory.createArrayBacked(ServerBlockEntityEvents.Load.class, callbacks -> (blockEntity, world) -> {
+		if (EventFactory.isProfilingEnabled()) {
+			final Profiler profiler = world.getProfiler();
+			profiler.push("fabricServerBlockEntityLoad");
+
+			for (ServerBlockEntityEvents.Load callback : callbacks) {
+				profiler.push(EventFactory.getHandlerName(callback));
+				callback.onLoad(blockEntity, world);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (ServerBlockEntityEvents.Load callback : callbacks) {
+				callback.onLoad(blockEntity, world);
+			}
+		}
+	});
+
+	/**
+	 * Called when an BlockEntity is about to be unloaded from a ServerWorld.
+	 *
+	 * <p>When this event is called, the block entity is still present on the world.
+	 */
+	public static final Event<Unload> BLOCK_ENTITY_UNLOAD = EventFactory.createArrayBacked(ServerBlockEntityEvents.Unload.class, callbacks -> (blockEntity, world) -> {
+		if (EventFactory.isProfilingEnabled()) {
+			final Profiler profiler = world.getProfiler();
+			profiler.push("fabricServerBlockEntityUnload");
+
+			for (ServerBlockEntityEvents.Unload callback : callbacks) {
+				profiler.push(EventFactory.getHandlerName(callback));
+				callback.onUnload(blockEntity, world);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (ServerBlockEntityEvents.Unload callback : callbacks) {
+				callback.onUnload(blockEntity, world);
+			}
+		}
+	});
+
+	public interface Load {
+		void onLoad(BlockEntity blockEntity, ServerWorld world);
+	}
+
+	public interface Unload {
+		void onUnload(BlockEntity blockEntity, ServerWorld world);
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/event/lifecycle/v1/ServerChunkEvents.java b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/event/lifecycle/v1/ServerChunkEvents.java
new file mode 100644
index 000000000..af495363e
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/event/lifecycle/v1/ServerChunkEvents.java
@@ -0,0 +1,85 @@
+/*
+ * 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.event.lifecycle.v1;
+
+import net.minecraft.server.world.ServerWorld;
+import net.minecraft.util.profiler.Profiler;
+import net.minecraft.world.chunk.WorldChunk;
+
+import net.fabricmc.fabric.api.event.Event;
+import net.fabricmc.fabric.api.event.EventFactory;
+
+public final class ServerChunkEvents {
+	private ServerChunkEvents() {
+	}
+
+	/**
+	 * Called when an chunk is loaded into a ServerWorld.
+	 *
+	 * <p>When this event is called, the chunk is already in the world.
+	 */
+	public static final Event<ServerChunkEvents.Load> CHUNK_LOAD = EventFactory.createArrayBacked(ServerChunkEvents.Load.class, callbacks -> (serverWorld, chunk) -> {
+		if (EventFactory.isProfilingEnabled()) {
+			final Profiler profiler = serverWorld.getProfiler();
+			profiler.push("fabricServerChunkLoad");
+
+			for (ServerChunkEvents.Load callback : callbacks) {
+				profiler.push(EventFactory.getHandlerName(callback));
+				callback.onChunkLoad(serverWorld, chunk);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (ServerChunkEvents.Load callback : callbacks) {
+				callback.onChunkLoad(serverWorld, chunk);
+			}
+		}
+	});
+
+	/**
+	 * Called when an chunk is unloaded from a ServerWorld.
+	 *
+	 * <p>When this event is called, the chunk is still present in the world.
+	 */
+	public static final Event<ServerChunkEvents.Unload> CHUNK_UNLOAD = EventFactory.createArrayBacked(ServerChunkEvents.Unload.class, callbacks -> (serverWorld, chunk) -> {
+		if (EventFactory.isProfilingEnabled()) {
+			final Profiler profiler = serverWorld.getProfiler();
+			profiler.push("fabricServerChunkUnload");
+
+			for (ServerChunkEvents.Unload callback : callbacks) {
+				profiler.push(EventFactory.getHandlerName(callback));
+				callback.onChunkUnload(serverWorld, chunk);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (ServerChunkEvents.Unload callback : callbacks) {
+				callback.onChunkUnload(serverWorld, chunk);
+			}
+		}
+	});
+
+	public interface Load {
+		void onChunkLoad(ServerWorld world, WorldChunk chunk);
+	}
+
+	public interface Unload {
+		void onChunkUnload(ServerWorld world, WorldChunk chunk);
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/event/lifecycle/v1/ServerEntityEvents.java b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/event/lifecycle/v1/ServerEntityEvents.java
new file mode 100644
index 000000000..9166bf0f4
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/event/lifecycle/v1/ServerEntityEvents.java
@@ -0,0 +1,59 @@
+/*
+ * 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.event.lifecycle.v1;
+
+import net.minecraft.entity.Entity;
+import net.minecraft.server.world.ServerWorld;
+import net.minecraft.util.profiler.Profiler;
+
+import net.fabricmc.fabric.api.event.Event;
+import net.fabricmc.fabric.api.event.EventFactory;
+
+public final class ServerEntityEvents {
+	private ServerEntityEvents() {
+	}
+
+	/**
+	 * Called when an Entity is loaded into a ServerWorld.
+	 *
+	 * <p>When this event is called, the entity is already in the world.
+	 *
+	 * <p>Note there is no corresponding unload event because entity unloads cannot be reliably tracked.
+	 */
+	public static final Event<ServerEntityEvents.Load> ENTITY_LOAD = EventFactory.createArrayBacked(ServerEntityEvents.Load.class, callbacks -> (entity, world) -> {
+		if (EventFactory.isProfilingEnabled()) {
+			final Profiler profiler = world.getProfiler();
+			profiler.push("fabricServerEntityLoad");
+
+			for (ServerEntityEvents.Load callback : callbacks) {
+				profiler.push(EventFactory.getHandlerName(callback));
+				callback.onLoad(entity, world);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (ServerEntityEvents.Load callback : callbacks) {
+				callback.onLoad(entity, world);
+			}
+		}
+	});
+
+	public interface Load {
+		void onLoad(Entity entity, ServerWorld world);
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/event/lifecycle/v1/ServerLifecycleEvents.java b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/event/lifecycle/v1/ServerLifecycleEvents.java
new file mode 100644
index 000000000..ce5fceba8
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/event/lifecycle/v1/ServerLifecycleEvents.java
@@ -0,0 +1,77 @@
+/*
+ * 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.event.lifecycle.v1;
+
+import net.minecraft.server.MinecraftServer;
+
+import net.fabricmc.fabric.api.event.Event;
+import net.fabricmc.fabric.api.event.EventFactory;
+
+public final class ServerLifecycleEvents {
+	private ServerLifecycleEvents() {
+	}
+
+	/**
+	 * Called when a Minecraft server has started and is about to tick for the first time.
+	 *
+	 * <p>At this stage, all worlds are live.
+	 */
+	public static final Event<ServerStarted> SERVER_STARTED = EventFactory.createArrayBacked(ServerStarted.class, (callbacks) -> (server) -> {
+		for (ServerStarted callback : callbacks) {
+			callback.onServerStarted(server);
+		}
+	});
+
+	/**
+	 * Called when a Minecraft server has started shutting down.
+	 * This occurs before the server's network channel is closed and before any players are disconnected.
+	 *
+	 * <p>For example, an integrated server will begin stopping, but it's client may continue to run.
+	 *
+	 * <p>All worlds are still present and can be modified.
+	 */
+	public static final Event<ServerStopping> SERVER_STOPPING = EventFactory.createArrayBacked(ServerStopping.class, (callbacks) -> (server) -> {
+		for (ServerStopping callback : callbacks) {
+			callback.onServerStopping(server);
+		}
+	});
+
+	/**
+	 * Called when a Minecraft server has stopped.
+	 * All worlds have been closed and all (block)entities and players have been unloaded.
+	 *
+	 * <p>For example, an {@link net.fabricmc.api.EnvType#CLIENT integrated server} will begin stopping, but it's client may continue to run.
+	 * Meanwhile for a {@link net.fabricmc.api.EnvType#SERVER dedicated server}, this will be the last event called.
+	 */
+	public static final Event<ServerStopped> SERVER_STOPPED = EventFactory.createArrayBacked(ServerStopped.class, callbacks -> server -> {
+		for (ServerStopped callback : callbacks) {
+			callback.onServerStopped(server);
+		}
+	});
+
+	public interface ServerStarted {
+		void onServerStarted(MinecraftServer server);
+	}
+
+	public interface ServerStopping {
+		void onServerStopping(MinecraftServer server);
+	}
+
+	public interface ServerStopped {
+		void onServerStopped(MinecraftServer server);
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/event/lifecycle/v1/ServerTickEvents.java b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/event/lifecycle/v1/ServerTickEvents.java
new file mode 100644
index 000000000..9b9d2662f
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/api/event/lifecycle/v1/ServerTickEvents.java
@@ -0,0 +1,135 @@
+/*
+ * 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.event.lifecycle.v1;
+
+import net.minecraft.server.MinecraftServer;
+import net.minecraft.server.world.ServerWorld;
+import net.minecraft.util.profiler.Profiler;
+
+import net.fabricmc.fabric.api.event.Event;
+import net.fabricmc.fabric.api.event.EventFactory;
+
+public final class ServerTickEvents {
+	private ServerTickEvents() {
+	}
+
+	/**
+	 * Called at the start of the server tick.
+	 */
+	public static final Event<StartTick> START_SERVER_TICK = EventFactory.createArrayBacked(StartTick.class, callbacks -> server -> {
+		if (EventFactory.isProfilingEnabled()) {
+			final Profiler profiler = server.getProfiler();
+			profiler.push("fabricStartServerTick");
+
+			for (StartTick event : callbacks) {
+				profiler.push(EventFactory.getHandlerName(event));
+				event.onStartTick(server);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (StartTick event : callbacks) {
+				event.onStartTick(server);
+			}
+		}
+	});
+
+	/**
+	 * Called at the end of the server tick.
+	 */
+	public static final Event<EndTick> END_SERVER_TICK = EventFactory.createArrayBacked(EndTick.class, callbacks -> server -> {
+		if (EventFactory.isProfilingEnabled()) {
+			final Profiler profiler = server.getProfiler();
+			profiler.push("fabricEndServerTick");
+
+			for (EndTick event : callbacks) {
+				profiler.push(EventFactory.getHandlerName(event));
+				event.onEndTick(server);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (EndTick event : callbacks) {
+				event.onEndTick(server);
+			}
+		}
+	});
+
+	/**
+	 * Called at the start of a ServerWorld's tick.
+	 */
+	public static final Event<StartWorldTick> START_WORLD_TICK = EventFactory.createArrayBacked(StartWorldTick.class, callbacks -> world -> {
+		if (EventFactory.isProfilingEnabled()) {
+			final Profiler profiler = world.getProfiler();
+			profiler.push("fabricStartServerWorldTick_" + world.getRegistryKey().getValue());
+
+			for (StartWorldTick callback : callbacks) {
+				profiler.push(EventFactory.getHandlerName(callback));
+				callback.onStartTick(world);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (StartWorldTick callback : callbacks) {
+				callback.onStartTick(world);
+			}
+		}
+	});
+
+	/**
+	 * Called at the end of a ServerWorld's tick.
+	 *
+	 * <p>End of world tick may be used to start async computations for the next tick.
+	 */
+	public static final Event<EndWorldTick> END_WORLD_TICK = EventFactory.createArrayBacked(EndWorldTick.class, callbacks -> world -> {
+		if (EventFactory.isProfilingEnabled()) {
+			final Profiler profiler = world.getProfiler();
+			profiler.push("fabricEndServerWorldTick_" + world.getRegistryKey().getValue());
+
+			for (EndWorldTick callback : callbacks) {
+				profiler.push(EventFactory.getHandlerName(callback));
+				callback.onEndTick(world);
+				profiler.pop();
+			}
+
+			profiler.pop();
+		} else {
+			for (EndWorldTick callback : callbacks) {
+				callback.onEndTick(world);
+			}
+		}
+	});
+
+	public interface StartTick {
+		void onStartTick(MinecraftServer server);
+	}
+
+	public interface EndTick {
+		void onEndTick(MinecraftServer server);
+	}
+
+	public interface StartWorldTick {
+		void onStartTick(ServerWorld world);
+	}
+
+	public interface EndWorldTick {
+		void onEndTick(ServerWorld world);
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/MinecraftServerMixin.java b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/MinecraftServerMixin.java
new file mode 100644
index 000000000..4d4cb4a92
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/MinecraftServerMixin.java
@@ -0,0 +1,76 @@
+/*
+ * 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.event.lifecycle;
+
+import java.util.Iterator;
+import java.util.List;
+import java.util.function.BooleanSupplier;
+
+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.callback.CallbackInfo;
+import org.spongepowered.asm.mixin.injection.callback.LocalCapture;
+
+import net.minecraft.block.entity.BlockEntity;
+import net.minecraft.entity.Entity;
+import net.minecraft.server.MinecraftServer;
+import net.minecraft.server.world.ServerWorld;
+
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerBlockEntityEvents;
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerLifecycleEvents;
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerTickEvents;
+
+@Mixin(MinecraftServer.class)
+public abstract class MinecraftServerMixin {
+	@Inject(at = @At(value = "INVOKE", target = "Lnet/minecraft/server/MinecraftServer;setFavicon(Lnet/minecraft/server/ServerMetadata;)V", ordinal = 0), method = "method_29741")
+	private void afterSetupServer(CallbackInfo info) {
+		ServerLifecycleEvents.SERVER_STARTED.invoker().onServerStarted((MinecraftServer) (Object) this);
+	}
+
+	@Inject(at = @At("HEAD"), method = "shutdown")
+	private void beforeShutdownServer(CallbackInfo info) {
+		ServerLifecycleEvents.SERVER_STOPPING.invoker().onServerStopping((MinecraftServer) (Object) this);
+	}
+
+	@Inject(at = @At("TAIL"), method = "shutdown")
+	private void afterShutdownServer(CallbackInfo info) {
+		ServerLifecycleEvents.SERVER_STOPPED.invoker().onServerStopped((MinecraftServer) (Object) this);
+	}
+
+	@Inject(at = @At(value = "INVOKE", target = "Lnet/minecraft/server/MinecraftServer;tickWorlds(Ljava/util/function/BooleanSupplier;)V"), method = "tick")
+	private void onStartTick(BooleanSupplier shouldKeepTicking, CallbackInfo ci) {
+		ServerTickEvents.START_SERVER_TICK.invoker().onStartTick((MinecraftServer) (Object) this);
+	}
+
+	@Inject(at = @At("TAIL"), method = "tick")
+	private void onEndTick(BooleanSupplier shouldKeepTicking, CallbackInfo info) {
+		ServerTickEvents.END_SERVER_TICK.invoker().onEndTick((MinecraftServer) (Object) this);
+	}
+
+	/**
+	 * When a world is closed, it means the world will be unloaded.
+	 */
+	@Inject(at = @At(value = "INVOKE", target = "Lnet/minecraft/server/world/ServerWorld;close()V"), method = "shutdown", locals = LocalCapture.CAPTURE_FAILEXCEPTION)
+	private void closeWorld(CallbackInfo ci, Iterator<ServerWorld> worlds, ServerWorld serverWorld) {
+		final List<Entity> entities = serverWorld.getEntities(null, entity -> true); // Get every single entity in the world
+
+		for (BlockEntity blockEntity : serverWorld.blockEntities) {
+			ServerBlockEntityEvents.BLOCK_ENTITY_UNLOAD.invoker().onUnload(blockEntity, serverWorld);
+		}
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/ServerWorldMixin.java b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/ServerWorldMixin.java
new file mode 100644
index 000000000..4c89416b9
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/ServerWorldMixin.java
@@ -0,0 +1,52 @@
+/*
+ * 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.event.lifecycle;
+
+import java.util.function.BooleanSupplier;
+
+import org.objectweb.asm.Opcodes;
+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.CallbackInfo;
+
+import net.minecraft.entity.Entity;
+import net.minecraft.server.world.ServerWorld;
+
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerEntityEvents;
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerTickEvents;
+
+@Mixin(ServerWorld.class)
+public abstract class ServerWorldMixin {
+	@Shadow
+	private boolean inEntityTick;
+
+	// Call our load event after vanilla has loaded the entity
+	@Inject(method = "loadEntityUnchecked", at = @At("TAIL"))
+	private void onLoadEntity(Entity entity, CallbackInfo ci) {
+		if (!this.inEntityTick) { // Copy vanilla logic, we cannot load entities while the game is ticking entities
+			ServerEntityEvents.ENTITY_LOAD.invoker().onLoad(entity, (ServerWorld) (Object) this);
+		}
+	}
+
+	// Make sure "insideBlockTick" is true before we call the start tick, so inject after it is set
+	@Inject(method = "tick", at = @At(value = "FIELD", target = "Lnet/minecraft/server/world/ServerWorld;inBlockTick:Z", opcode = Opcodes.PUTFIELD, ordinal = 0, shift = At.Shift.AFTER))
+	private void startWorldTick(BooleanSupplier shouldKeepTicking, CallbackInfo ci) {
+		ServerTickEvents.START_WORLD_TICK.invoker().onStartTick((ServerWorld) (Object) this);
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/ThreadedAnvilChunkStorageMixin.java b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/ThreadedAnvilChunkStorageMixin.java
new file mode 100644
index 000000000..76209c9b4
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/ThreadedAnvilChunkStorageMixin.java
@@ -0,0 +1,68 @@
+/*
+ * 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.event.lifecycle;
+
+import java.util.concurrent.CompletableFuture;
+
+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.CallbackInfo;
+import org.spongepowered.asm.mixin.injection.callback.CallbackInfoReturnable;
+
+import net.minecraft.server.world.ChunkHolder;
+import net.minecraft.server.world.ServerWorld;
+import net.minecraft.server.world.ThreadedAnvilChunkStorage;
+import net.minecraft.world.chunk.Chunk;
+import net.minecraft.world.chunk.WorldChunk;
+
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerChunkEvents;
+
+@Mixin(ThreadedAnvilChunkStorage.class)
+public abstract class ThreadedAnvilChunkStorageMixin {
+	@Shadow
+	@Final
+	private ServerWorld world;
+
+	// Chunk (Un)Load events, An explanation:
+	// Must of this code is wrapped inside of futures and consumers, so it's generally a mess.
+
+	/**
+	 * Injection is inside of tryUnloadChunk.
+	 * We inject just after "setLoadedToWorld" is made false, since here the WorldChunk is guaranteed to be unloaded.
+	 */
+	@Inject(method = "method_18843", at = @At(value = "INVOKE", target = "Lnet/minecraft/world/chunk/WorldChunk;setLoadedToWorld(Z)V", shift = At.Shift.AFTER))
+	private void onChunkUnload(ChunkHolder chunkHolder, CompletableFuture<Chunk> chunkFuture, long pos, Chunk chunk, CallbackInfo ci) {
+		ServerChunkEvents.CHUNK_UNLOAD.invoker().onChunkUnload(this.world, (WorldChunk) chunk);
+	}
+
+	/**
+	 * Injection is inside of convertToFullChunk?
+	 *
+	 * <p>The following is expected contractually
+	 *
+	 * <ul><li>the chunk being loaded MUST be a WorldChunk.
+	 * <li>everything within the chunk has been loaded into the world. Entities, BlockEntities, etc.</ul>
+	 */
+	@Inject(method = "method_17227", at = @At("TAIL"))
+	private void onChunkLoad(ChunkHolder chunkHolder, Chunk protoChunk, CallbackInfoReturnable<Chunk> callbackInfoReturnable) {
+		// We fire the event at TAIL since the chunk is guaranteed to be a WorldChunk then.
+		ServerChunkEvents.CHUNK_LOAD.invoker().onChunkLoad(this.world, (WorldChunk) callbackInfoReturnable.getReturnValue());
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/WorldMixin.java b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/WorldMixin.java
new file mode 100644
index 000000000..f4f094142
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/WorldMixin.java
@@ -0,0 +1,90 @@
+/*
+ * 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.event.lifecycle;
+
+import java.util.Collection;
+import java.util.Iterator;
+import java.util.List;
+
+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.Redirect;
+import org.spongepowered.asm.mixin.injection.Slice;
+import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
+import org.spongepowered.asm.mixin.injection.callback.CallbackInfoReturnable;
+import org.spongepowered.asm.mixin.injection.callback.LocalCapture;
+
+import net.minecraft.block.entity.BlockEntity;
+import net.minecraft.server.world.ServerWorld;
+import net.minecraft.util.math.BlockPos;
+import net.minecraft.util.profiler.Profiler;
+import net.minecraft.world.World;
+
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerBlockEntityEvents;
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerTickEvents;
+
+@Mixin(World.class)
+public abstract class WorldMixin {
+	@Shadow
+	public abstract boolean isClient();
+
+	@Shadow
+	public abstract Profiler getProfiler();
+
+	@Inject(method = "addBlockEntity", at = @At("TAIL"))
+	protected void onLoadBlockEntity(BlockEntity blockEntity, CallbackInfoReturnable<Boolean> cir) {
+		if (!this.isClient()) { // Only fire this event if we are a server world
+			ServerBlockEntityEvents.BLOCK_ENTITY_LOAD.invoker().onLoad(blockEntity, (ServerWorld) (Object) this);
+		}
+	}
+
+	// Mojang what hell, why do you need three ways to unload block entities
+	@Inject(method = "removeBlockEntity", at = @At(value = "INVOKE", target = "Ljava/util/List;remove(Ljava/lang/Object;)Z", ordinal = 1), locals = LocalCapture.CAPTURE_FAILEXCEPTION)
+	protected void onUnloadBlockEntity(BlockPos pos, CallbackInfo ci, BlockEntity blockEntity) {
+		if (!this.isClient()) { // Only fire this event if we are a server world
+			ServerBlockEntityEvents.BLOCK_ENTITY_UNLOAD.invoker().onUnload(blockEntity, (ServerWorld) (Object) this);
+		}
+	}
+
+	@Inject(method = "tickBlockEntities", at = @At(value = "INVOKE", target = "Ljava/util/List;remove(Ljava/lang/Object;)Z"), slice = @Slice(from = @At(value = "INVOKE", target = "Lnet/minecraft/util/profiler/Profiler;pop()V"), to = @At(value = "INVOKE", target = "Lnet/minecraft/world/chunk/WorldChunk;removeBlockEntity(Lnet/minecraft/util/math/BlockPos;)V")), locals = LocalCapture.CAPTURE_FAILEXCEPTION)
+	protected void onRemoveBlockEntity(CallbackInfo ci, Profiler profiler, Iterator iterator, BlockEntity blockEntity) {
+		if (!this.isClient()) {
+			ServerBlockEntityEvents.BLOCK_ENTITY_UNLOAD.invoker().onUnload(blockEntity, (ServerWorld) (Object) this);
+		}
+	}
+
+	@Redirect(method = "tickBlockEntities", at = @At(value = "INVOKE", target = "Ljava/util/List;removeAll(Ljava/util/Collection;)Z", ordinal = 1))
+	protected boolean onPurgeRemovedBlockEntities(List<BlockEntity> blockEntityList, Collection<BlockEntity> removals) {
+		if (!this.isClient()) {
+			for (BlockEntity removal : removals) {
+				ServerBlockEntityEvents.BLOCK_ENTITY_UNLOAD.invoker().onUnload(removal, (ServerWorld) (Object) this);
+			}
+		}
+
+		// Mimic vanilla logic
+		return blockEntityList.removeAll(removals);
+	}
+
+	@Inject(at = @At("RETURN"), method = "tickBlockEntities")
+	protected void tickWorldAfterBlockEntities(CallbackInfo ci) {
+		if (!this.isClient()) {
+			ServerTickEvents.END_WORLD_TICK.invoker().onEndTick((ServerWorld) (Object) this);
+		}
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/client/ClientChunkManagerMixin.java b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/client/ClientChunkManagerMixin.java
new file mode 100644
index 000000000..0dfa1d2c8
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/client/ClientChunkManagerMixin.java
@@ -0,0 +1,55 @@
+/*
+ * 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.event.lifecycle.client;
+
+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.CallbackInfo;
+import org.spongepowered.asm.mixin.injection.callback.CallbackInfoReturnable;
+import org.spongepowered.asm.mixin.injection.callback.LocalCapture;
+
+import net.minecraft.client.world.ClientChunkManager;
+import net.minecraft.client.world.ClientWorld;
+import net.minecraft.nbt.CompoundTag;
+import net.minecraft.network.PacketByteBuf;
+import net.minecraft.world.biome.source.BiomeArray;
+import net.minecraft.world.chunk.WorldChunk;
+
+import net.fabricmc.api.EnvType;
+import net.fabricmc.api.Environment;
+import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientChunkEvents;
+
+@Environment(EnvType.CLIENT)
+@Mixin(ClientChunkManager.class)
+public abstract class ClientChunkManagerMixin {
+	@Final
+	@Shadow
+	private ClientWorld world;
+
+	@Inject(method = "loadChunkFromPacket", at = @At("TAIL")) // 1.16 has a boolean param here. I think it means whether the packet is complete.
+	private void onChunkLoad(int chunkX, int chunkZ, BiomeArray biomes, PacketByteBuf buf, CompoundTag tag, int k, boolean complete, CallbackInfoReturnable<WorldChunk> cir) {
+		ClientChunkEvents.CHUNK_LOAD.invoker().onChunkLoad(this.world, cir.getReturnValue());
+	}
+
+	@Inject(method = "unload", at = @At(value = "INVOKE", target = "Lnet/minecraft/client/world/ClientChunkManager$ClientChunkMap;compareAndSet(ILnet/minecraft/world/chunk/WorldChunk;Lnet/minecraft/world/chunk/WorldChunk;)Lnet/minecraft/world/chunk/WorldChunk;", shift = At.Shift.AFTER), locals = LocalCapture.CAPTURE_FAILEXCEPTION)
+	private void onChunkUnload(int chunkX, int chunkZ, CallbackInfo ci, int i, WorldChunk chunk) {
+		ClientChunkEvents.CHUNK_UNLOAD.invoker().onChunkUnload(this.world, chunk);
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/client/ClientPlayNetworkHandlerMixin.java b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/client/ClientPlayNetworkHandlerMixin.java
new file mode 100644
index 000000000..712e3d367
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/client/ClientPlayNetworkHandlerMixin.java
@@ -0,0 +1,94 @@
+/*
+ * 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.event.lifecycle.client;
+
+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.CallbackInfo;
+
+import net.minecraft.block.entity.BlockEntity;
+import net.minecraft.client.network.ClientPlayNetworkHandler;
+import net.minecraft.client.world.ClientWorld;
+import net.minecraft.entity.Entity;
+import net.minecraft.network.packet.s2c.play.GameJoinS2CPacket;
+import net.minecraft.network.packet.s2c.play.PlayerRespawnS2CPacket;
+
+import net.fabricmc.api.EnvType;
+import net.fabricmc.api.Environment;
+import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientBlockEntityEvents;
+import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientEntityEvents;
+
+@Environment(EnvType.CLIENT)
+@Mixin(ClientPlayNetworkHandler.class)
+public abstract class ClientPlayNetworkHandlerMixin {
+	@Shadow
+	private ClientWorld world;
+
+	@Inject(method = "onPlayerRespawn", at = @At(value = "NEW", target = "net/minecraft/client/world/ClientWorld"))
+	private void onPlayerRespawn(PlayerRespawnS2CPacket packet, CallbackInfo ci) {
+		// If a world already exists, we need to unload all (block)entities in the world.
+		if (this.world != null) {
+			for (Entity entity : world.getEntities()) {
+				ClientEntityEvents.ENTITY_UNLOAD.invoker().onUnload(entity, this.world);
+			}
+
+			for (BlockEntity blockEntity : world.blockEntities) {
+				ClientBlockEntityEvents.BLOCK_ENTITY_UNLOAD.invoker().onUnload(blockEntity, this.world);
+				// No need to clear the `tickingBlockEntities` list since it will be null in just an instant
+			}
+		}
+	}
+
+	/**
+	 * An explanation why we unload entities during onGameJoin:
+	 * Proxies such as Waterfall may send another Game Join packet if entity meta rewrite is disabled, so we will cover ourselves.
+	 * Velocity by default will send a Game Join packet when the player changes servers, which will create a new client world.
+	 * Also anyone can send another GameJoinPacket at any time, so we need to watch out.
+	 */
+	@Inject(method = "onGameJoin", at = @At(value = "NEW", target = "net/minecraft/client/world/ClientWorld"))
+	private void onGameJoin(GameJoinS2CPacket packet, CallbackInfo ci) {
+		// If a world already exists, we need to unload all (block)entities in the world.
+		if (this.world != null) {
+			for (Entity entity : world.getEntities()) {
+				ClientEntityEvents.ENTITY_UNLOAD.invoker().onUnload(entity, this.world);
+			}
+
+			for (BlockEntity blockEntity : world.blockEntities) {
+				ClientBlockEntityEvents.BLOCK_ENTITY_UNLOAD.invoker().onUnload(blockEntity, this.world);
+				// No need to clear the `tickingBlockEntities` list since it will be null in just an instant
+			}
+		}
+	}
+
+	// Called when the client disconnects from a server.
+	@Inject(method = "clearWorld", at = @At("HEAD"))
+	private void onClearWorld(CallbackInfo ci) {
+		// If a world already exists, we need to unload all (block)entities in the world.
+		if (this.world != null) {
+			for (Entity entity : world.getEntities()) {
+				ClientEntityEvents.ENTITY_UNLOAD.invoker().onUnload(entity, this.world);
+			}
+
+			for (BlockEntity blockEntity : world.blockEntities) {
+				ClientBlockEntityEvents.BLOCK_ENTITY_UNLOAD.invoker().onUnload(blockEntity, this.world);
+				// No need to clear the `tickingBlockEntities` list since it will be null in just an instant
+			}
+		}
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/client/ClientWorldMixin.java b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/client/ClientWorldMixin.java
new file mode 100644
index 000000000..6ad6af0ce
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/client/ClientWorldMixin.java
@@ -0,0 +1,93 @@
+/*
+ * 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.event.lifecycle.client;
+
+import java.util.Collection;
+import java.util.Iterator;
+import java.util.List;
+
+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.callback.CallbackInfo;
+import org.spongepowered.asm.mixin.injection.callback.CallbackInfoReturnable;
+
+import net.minecraft.block.entity.BlockEntity;
+import net.minecraft.client.world.ClientWorld;
+import net.minecraft.entity.Entity;
+import net.minecraft.util.math.BlockPos;
+import net.minecraft.util.profiler.Profiler;
+
+import net.fabricmc.api.EnvType;
+import net.fabricmc.api.Environment;
+import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientBlockEntityEvents;
+import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientEntityEvents;
+import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientTickEvents;
+import net.fabricmc.fabric.mixin.event.lifecycle.WorldMixin;
+
+@Environment(EnvType.CLIENT)
+@Mixin(ClientWorld.class)
+public abstract class ClientWorldMixin extends WorldMixin {
+	// Call our load event after vanilla has loaded the entity
+	@Inject(method = "addEntityPrivate", at = @At("TAIL"))
+	private void onEntityLoad(int id, Entity entity, CallbackInfo ci) {
+		ClientEntityEvents.ENTITY_LOAD.invoker().onLoad(entity, (ClientWorld) (Object) this);
+	}
+
+	// Call our unload event before vanilla does.
+	@Inject(method = "finishRemovingEntity", at = @At("HEAD"))
+	private void onEntityUnload(Entity entity, CallbackInfo ci) {
+		ClientEntityEvents.ENTITY_UNLOAD.invoker().onUnload(entity, (ClientWorld) (Object) this);
+	}
+
+	// We override our injection on the clientworld so only the client's block entity invocations will run
+	@Override
+	protected void onLoadBlockEntity(BlockEntity blockEntity, CallbackInfoReturnable<Boolean> cir) {
+		ClientBlockEntityEvents.BLOCK_ENTITY_LOAD.invoker().onLoad(blockEntity, (ClientWorld) (Object) this);
+	}
+
+	// We override our injection on the clientworld so only the client's block entity invocations will run
+	@Override
+	protected void onUnloadBlockEntity(BlockPos pos, CallbackInfo ci, BlockEntity blockEntity) {
+		ClientBlockEntityEvents.BLOCK_ENTITY_UNLOAD.invoker().onUnload(blockEntity, (ClientWorld) (Object) this);
+	}
+
+	@Override
+	protected void onRemoveBlockEntity(CallbackInfo ci, Profiler profiler, Iterator iterator, BlockEntity blockEntity) {
+		ClientBlockEntityEvents.BLOCK_ENTITY_UNLOAD.invoker().onUnload(blockEntity, (ClientWorld) (Object) this);
+	}
+
+	@Override
+	protected boolean onPurgeRemovedBlockEntities(List<BlockEntity> blockEntityList, Collection<BlockEntity> removals) {
+		for (BlockEntity removal : removals) {
+			ClientBlockEntityEvents.BLOCK_ENTITY_UNLOAD.invoker().onUnload(removal, (ClientWorld) (Object) this);
+		}
+
+		return super.onPurgeRemovedBlockEntities(blockEntityList, removals); // Call super
+	}
+
+	// We override our injection on the clientworld so only the client world's tick invocations will run
+	@Override
+	protected void tickWorldAfterBlockEntities(CallbackInfo ci) {
+		ClientTickEvents.END_WORLD_TICK.invoker().onEndTick((ClientWorld) (Object) this);
+	}
+
+	@Inject(method = "tickEntities", at = @At("HEAD"))
+	private void startWorldTick(CallbackInfo ci) {
+		ClientTickEvents.START_WORLD_TICK.invoker().onStartTick((ClientWorld) (Object) this);
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/client/MinecraftClientMixin.java b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/client/MinecraftClientMixin.java
new file mode 100644
index 000000000..588f64961
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/java/net/fabricmc/fabric/mixin/event/lifecycle/client/MinecraftClientMixin.java
@@ -0,0 +1,54 @@
+/*
+ * 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.event.lifecycle.client;
+
+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.callback.CallbackInfo;
+
+import net.minecraft.client.MinecraftClient;
+
+import net.fabricmc.api.EnvType;
+import net.fabricmc.api.Environment;
+import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientTickEvents;
+import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientLifecycleEvents;
+
+@Environment(EnvType.CLIENT)
+@Mixin(MinecraftClient.class)
+public abstract class MinecraftClientMixin {
+	@Inject(at = @At("HEAD"), method = "tick")
+	private void onStartTick(CallbackInfo info) {
+		ClientTickEvents.START_CLIENT_TICK.invoker().onStartTick((MinecraftClient) (Object) this);
+	}
+
+	@Inject(at = @At("RETURN"), method = "tick")
+	private void onEndTick(CallbackInfo info) {
+		ClientTickEvents.END_CLIENT_TICK.invoker().onEndTick((MinecraftClient) (Object) this);
+	}
+
+	@Inject(at = @At(value = "INVOKE", target = "Lorg/apache/logging/log4j/Logger;info(Ljava/lang/String;)V", shift = At.Shift.AFTER), method = "stop")
+	private void onStopping(CallbackInfo ci) {
+		ClientLifecycleEvents.CLIENT_STOPPING.invoker().onClientStopping((MinecraftClient) (Object) this);
+	}
+
+	// We inject after the thread field is set so `ThreadExecutor#getThread` will work
+	@Inject(at = @At(value = "FIELD", target = "Lnet/minecraft/client/MinecraftClient;thread:Ljava/lang/Thread;", shift = At.Shift.AFTER), method = "run")
+	private void onStart(CallbackInfo ci) {
+		ClientLifecycleEvents.CLIENT_STARTED.invoker().onClientStarted((MinecraftClient) (Object) this);
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/main/resources/assets/fabric-lifecycle-events-v1/icon.png b/fabric-lifecycle-events-v1/src/main/resources/assets/fabric-lifecycle-events-v1/icon.png
new file mode 100644
index 000000000..2931efbf6
Binary files /dev/null and b/fabric-lifecycle-events-v1/src/main/resources/assets/fabric-lifecycle-events-v1/icon.png differ
diff --git a/fabric-lifecycle-events-v1/src/main/resources/fabric-lifecycle-events-v1.mixins.json b/fabric-lifecycle-events-v1/src/main/resources/fabric-lifecycle-events-v1.mixins.json
new file mode 100644
index 000000000..a41feb4e1
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/resources/fabric-lifecycle-events-v1.mixins.json
@@ -0,0 +1,20 @@
+{
+  "required": true,
+  "package": "net.fabricmc.fabric.mixin.event.lifecycle",
+  "compatibilityLevel": "JAVA_8",
+  "mixins": [
+    "MinecraftServerMixin",
+    "ServerWorldMixin",
+    "ThreadedAnvilChunkStorageMixin",
+    "WorldMixin"
+  ],
+  "client": [
+    "client.ClientChunkManagerMixin",
+    "client.ClientPlayNetworkHandlerMixin",
+    "client.ClientWorldMixin",
+    "client.MinecraftClientMixin"
+  ],
+  "injectors": {
+    "defaultRequire": 1
+  }
+}
diff --git a/fabric-lifecycle-events-v1/src/main/resources/fabric.mod.json b/fabric-lifecycle-events-v1/src/main/resources/fabric.mod.json
new file mode 100644
index 000000000..f0c3745e2
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/main/resources/fabric.mod.json
@@ -0,0 +1,26 @@
+{
+  "schemaVersion": 1,
+  "id": "fabric-lifecycle-events-v1",
+  "name": "Fabric Lifecycle Events (v1)",
+  "version": "${version}",
+  "environment": "*",
+  "license": "Apache-2.0",
+  "icon": "assets/fabric-lifecycle-events-v1/icon.png",
+  "contact": {
+    "homepage": "https://fabricmc.net",
+    "irc": "irc://irc.esper.net:6667/fabric",
+    "issues": "https://github.com/FabricMC/fabric/issues",
+    "sources": "https://github.com/FabricMC/fabric"
+  },
+  "authors": [
+    "FabricMC"
+  ],
+  "mixins": [
+    "fabric-lifecycle-events-v1.mixins.json"
+  ],
+  "depends": {
+    "fabricloader": ">=0.4.0",
+    "fabric-api-base": "*"
+  },
+  "description": "Events for the game's lifecycle."
+}
diff --git a/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/ServerBlockEntityLifecycleTests.java b/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/ServerBlockEntityLifecycleTests.java
new file mode 100644
index 000000000..d9e0d6b5f
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/ServerBlockEntityLifecycleTests.java
@@ -0,0 +1,77 @@
+/*
+ * 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.test.event.lifecycle;
+
+import java.util.ArrayList;
+import java.util.List;
+
+import org.apache.logging.log4j.Logger;
+
+import net.minecraft.block.entity.BlockEntity;
+import net.minecraft.server.world.ServerWorld;
+import net.minecraft.util.registry.Registry;
+
+import net.fabricmc.api.ModInitializer;
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerBlockEntityEvents;
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerLifecycleEvents;
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerTickEvents;
+
+public class ServerBlockEntityLifecycleTests implements ModInitializer {
+	private List<BlockEntity> serverBlockEntities = new ArrayList<>();
+
+	@Override
+	public void onInitialize() {
+		final Logger logger = ServerLifecycleTests.LOGGER;
+
+		ServerBlockEntityEvents.BLOCK_ENTITY_LOAD.register((blockEntity, world) -> {
+			this.serverBlockEntities.add(blockEntity);
+			logger.info("[SERVER] LOADED " + Registry.BLOCK_ENTITY_TYPE.getId(blockEntity.getType()).toString() + " - BlockEntities: " + this.serverBlockEntities.size());
+		});
+
+		ServerBlockEntityEvents.BLOCK_ENTITY_UNLOAD.register((blockEntity, world) -> {
+			this.serverBlockEntities.remove(blockEntity);
+			logger.info("[SERVER] UNLOADED " + Registry.BLOCK_ENTITY_TYPE.getId(blockEntity.getType()).toString() + " - BlockEntities: " + this.serverBlockEntities.size());
+		});
+
+		ServerTickEvents.END_SERVER_TICK.register(minecraftServer -> {
+			if (minecraftServer.getTicks() % 200 == 0) {
+				int entities = 0;
+				logger.info("[SERVER] Tracked BlockEntities:" + this.serverBlockEntities.size() + " Ticked at: " + minecraftServer.getTicks() + "ticks");
+
+				for (ServerWorld world : minecraftServer.getWorlds()) {
+					int worldEntities = world.blockEntities.size();
+					logger.info("[SERVER] Tracked BlockEntities in " + world.getRegistryKey().toString() + " - " + worldEntities);
+					entities += worldEntities;
+				}
+
+				logger.info("[SERVER] Actual Total BlockEntities: " + entities);
+
+				if (entities != this.serverBlockEntities.size()) {
+					logger.error("[SERVER] Mismatch in tracked blockentities and actual blockentities");
+				}
+			}
+		});
+
+		ServerLifecycleEvents.SERVER_STOPPED.register(minecraftServer -> {
+			logger.info("[SERVER] Disconnected. Tracking: " + this.serverBlockEntities.size() + " blockentities");
+
+			if (this.serverBlockEntities.size() != 0) {
+				logger.error("[SERVER] Mismatch in tracked blockentities, expected 0");
+			}
+		});
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/ServerEntityLifecycleTests.java b/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/ServerEntityLifecycleTests.java
new file mode 100644
index 000000000..4bf9c0e47
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/ServerEntityLifecycleTests.java
@@ -0,0 +1,44 @@
+/*
+ * 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.test.event.lifecycle;
+
+import java.util.ArrayList;
+import java.util.List;
+
+import org.apache.logging.log4j.Logger;
+
+import net.minecraft.entity.Entity;
+
+import net.fabricmc.api.ModInitializer;
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerEntityEvents;
+
+/**
+ * Tests related to the lifecycle of entities.
+ */
+public class ServerEntityLifecycleTests implements ModInitializer {
+	private List<Entity> serverEntities = new ArrayList<>();
+
+	@Override
+	public void onInitialize() {
+		final Logger logger = ServerLifecycleTests.LOGGER;
+
+		ServerEntityEvents.ENTITY_LOAD.register((entity, world) -> {
+			this.serverEntities.add(entity);
+			logger.info("[SERVER] LOADED " + entity.toString() + " - Entities: " + this.serverEntities.size());
+		});
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/ServerLifecycleTests.java b/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/ServerLifecycleTests.java
new file mode 100644
index 000000000..0e0888cce
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/ServerLifecycleTests.java
@@ -0,0 +1,45 @@
+/*
+ * 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.test.event.lifecycle;
+
+import org.apache.logging.log4j.LogManager;
+import org.apache.logging.log4j.Logger;
+
+import net.fabricmc.api.ModInitializer;
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerLifecycleEvents;
+
+/**
+ * Tests related to the lifecycle of a server.
+ */
+public class ServerLifecycleTests implements ModInitializer {
+	public static final Logger LOGGER = LogManager.getLogger("LifecycleEventsTest");
+
+	@Override
+	public void onInitialize() {
+		ServerLifecycleEvents.SERVER_STARTED.register(server -> {
+			LOGGER.info("Started Server!");
+		});
+
+		ServerLifecycleEvents.SERVER_STOPPING.register(server -> {
+			LOGGER.info("Stopping Server!");
+		});
+
+		ServerLifecycleEvents.SERVER_STOPPED.register(server -> {
+			LOGGER.info("Stopped Server!");
+		});
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/ServerTickTests.java b/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/ServerTickTests.java
new file mode 100644
index 000000000..a0ca7a3f9
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/ServerTickTests.java
@@ -0,0 +1,59 @@
+/*
+ * 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.test.event.lifecycle;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import net.minecraft.util.registry.RegistryKey;
+import net.minecraft.world.World;
+
+import net.fabricmc.api.ModInitializer;
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerTickEvents;
+
+/**
+ * Test related to ticking events on the server.
+ */
+public class ServerTickTests implements ModInitializer {
+	private Map<RegistryKey<World>, Integer> tickTracker = new HashMap<>();
+
+	@Override
+	public void onInitialize() {
+		ServerTickEvents.END_SERVER_TICK.register(server -> {
+			if (server.getTicks() % 200 == 0) { // Log every 200 ticks to verify the tick callback works on the server
+				ServerLifecycleTests.LOGGER.info("Ticked Server at " + server.getTicks() + " ticks.");
+			}
+		});
+
+		ServerTickEvents.START_WORLD_TICK.register(world -> {
+			// Verify we are inside the tick
+			if (!world.isInBlockTick()) {
+				throw new AssertionError("Start tick event should be fired while ServerWorld is inside of block tick");
+			}
+		});
+
+		ServerTickEvents.END_WORLD_TICK.register(world -> {
+			final int worldTicks = tickTracker.computeIfAbsent(world.getRegistryKey(), k -> 0);
+
+			if (worldTicks % 200 == 0) { // Log every 200 ticks to verify the tick callback works on the server world
+				ServerLifecycleTests.LOGGER.info("Ticked Server World - " + worldTicks + " ticks:" + world.getRegistryKey().getValue());
+			}
+
+			this.tickTracker.put(world.getRegistryKey(), worldTicks + 1);
+		});
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/client/ClientBlockEntityLifecycleTests.java b/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/client/ClientBlockEntityLifecycleTests.java
new file mode 100644
index 000000000..51424dc60
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/client/ClientBlockEntityLifecycleTests.java
@@ -0,0 +1,73 @@
+/*
+ * 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.test.event.lifecycle.client;
+
+import java.util.ArrayList;
+import java.util.List;
+
+import org.apache.logging.log4j.Logger;
+
+import net.minecraft.block.entity.BlockEntity;
+import net.minecraft.util.registry.Registry;
+
+import net.fabricmc.api.ClientModInitializer;
+import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientBlockEntityEvents;
+import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientTickEvents;
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerLifecycleEvents;
+import net.fabricmc.fabric.test.event.lifecycle.ServerLifecycleTests;
+
+public class ClientBlockEntityLifecycleTests implements ClientModInitializer {
+	private List<BlockEntity> clientBlockEntities = new ArrayList<>();
+	private int clientTicks;
+
+	@Override
+	public void onInitializeClient() {
+		final Logger logger = ServerLifecycleTests.LOGGER;
+
+		ClientBlockEntityEvents.BLOCK_ENTITY_LOAD.register((blockEntity, world) -> {
+			this.clientBlockEntities.add(blockEntity);
+			logger.info("[CLIENT]" + " LOADED " + Registry.BLOCK_ENTITY_TYPE.getId(blockEntity.getType()).toString() + " - BlockEntities: " + this.clientBlockEntities.size());
+		});
+
+		ClientBlockEntityEvents.BLOCK_ENTITY_UNLOAD.register((blockEntity, world) -> {
+			this.clientBlockEntities.remove(blockEntity);
+			logger.info("[CLIENT]" + " UNLOADED " + Registry.BLOCK_ENTITY_TYPE.getId(blockEntity.getType()).toString() + " - BlockEntities: " + this.clientBlockEntities.size());
+		});
+
+		ClientTickEvents.END_CLIENT_TICK.register(client -> {
+			if (this.clientTicks++ % 200 == 0 && client.world != null) {
+				final int blockEntities = client.world.blockEntities.size();
+				logger.info("[CLIENT] Tracked BlockEntities:" + this.clientBlockEntities.size() + " Ticked at: " + this.clientTicks + "ticks");
+				logger.info("[CLIENT] Actual BlockEntities: " + client.world.blockEntities.size());
+
+				if (blockEntities != this.clientBlockEntities.size()) {
+					logger.error("[CLIENT] Mismatch in tracked blockentities and actual blockentities");
+				}
+			}
+		});
+
+		ServerLifecycleEvents.SERVER_STOPPED.register(minecraftServer -> {
+			if (!minecraftServer.isDedicated()) { // fixme: Use ClientNetworking#PLAY_DISCONNECTED instead of the server stop callback for testing.
+				logger.info("[CLIENT] Disconnected. Tracking: " + this.clientBlockEntities.size() + " blockentities");
+
+				if (this.clientBlockEntities.size() != 0) {
+					logger.error("[CLIENT] Mismatch in tracked blockentities, expected 0");
+				}
+			}
+		});
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/client/ClientEntityLifecycleTests.java b/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/client/ClientEntityLifecycleTests.java
new file mode 100644
index 000000000..350ba0709
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/client/ClientEntityLifecycleTests.java
@@ -0,0 +1,80 @@
+/*
+ * 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.test.event.lifecycle.client;
+
+import java.util.ArrayList;
+import java.util.List;
+
+import com.google.common.collect.Iterables;
+import org.apache.logging.log4j.Logger;
+
+import net.minecraft.entity.Entity;
+import net.minecraft.util.registry.Registry;
+
+import net.fabricmc.api.ClientModInitializer;
+import net.fabricmc.api.EnvType;
+import net.fabricmc.api.Environment;
+import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientEntityEvents;
+import net.fabricmc.fabric.api.event.lifecycle.v1.ServerLifecycleEvents;
+import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientTickEvents;
+import net.fabricmc.fabric.test.event.lifecycle.ServerLifecycleTests;
+
+/**
+ * Tests related to the lifecycle of entities.
+ */
+@Environment(EnvType.CLIENT)
+public class ClientEntityLifecycleTests implements ClientModInitializer {
+	private List<Entity> clientEntities = new ArrayList<>();
+	private int clientTicks;
+
+	@Override
+	public void onInitializeClient() {
+		final Logger logger = ServerLifecycleTests.LOGGER;
+
+		ClientEntityEvents.ENTITY_LOAD.register((entity, world) -> {
+			this.clientEntities.add(entity);
+			logger.info("[CLIENT]" + " LOADED " + Registry.ENTITY_TYPE.getId(entity.getType()).toString() + " - Entities: " + this.clientEntities.size());
+		});
+
+		ClientEntityEvents.ENTITY_UNLOAD.register((entity, world) -> {
+			this.clientEntities.remove(entity);
+			logger.info("[CLIENT]" + " UNLOADED " + Registry.ENTITY_TYPE.getId(entity.getType()).toString() + " - Entities: " + this.clientEntities.size());
+		});
+
+		ClientTickEvents.END_CLIENT_TICK.register(client -> {
+			if (this.clientTicks++ % 200 == 0 && client.world != null) {
+				final int entities = Iterables.toArray(client.world.getEntities(), Entity.class).length;
+				logger.info("[CLIENT] Tracked Entities:" + this.clientEntities.size() + " Ticked at: " + this.clientTicks + "ticks");
+				logger.info("[CLIENT] Actual Entities: " + entities);
+
+				if (entities != this.clientEntities.size()) {
+					logger.error("[CLIENT] Mismatch in tracked entities and actual entities");
+				}
+			}
+		});
+
+		ServerLifecycleEvents.SERVER_STOPPED.register(minecraftServer -> {
+			if (!minecraftServer.isDedicated()) { // fixme: Use ClientNetworking#PLAY_DISCONNECTED instead of the server stop callback for testing.
+				logger.info("[CLIENT] Disconnected. Tracking: " + this.clientEntities.size() + " entities");
+
+				if (this.clientEntities.size() != 0) {
+					logger.error("[CLIENT] Mismatch in tracked entities, expected 0");
+				}
+			}
+		});
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/client/ClientLifecycleTests.java b/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/client/ClientLifecycleTests.java
new file mode 100644
index 000000000..f649a0682
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/client/ClientLifecycleTests.java
@@ -0,0 +1,38 @@
+/*
+ * 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.test.event.lifecycle.client;
+
+import net.fabricmc.api.ClientModInitializer;
+import net.fabricmc.api.EnvType;
+import net.fabricmc.api.Environment;
+import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientLifecycleEvents;
+
+@Environment(EnvType.CLIENT)
+public class ClientLifecycleTests implements ClientModInitializer {
+	@Override
+	public void onInitializeClient() {
+		ClientLifecycleEvents.CLIENT_STARTED.register(client -> {
+			client.submitAndJoin(() -> { // This should fail if the client thread was not bound yet.
+				System.out.println("Started the client");
+			});
+		});
+
+		ClientLifecycleEvents.CLIENT_STOPPING.register(client -> {
+			System.out.println("Client has started stopping!");
+		});
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/client/ClientTickTests.java b/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/client/ClientTickTests.java
new file mode 100644
index 000000000..b033f992f
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/testmod/java/net/fabricmc/fabric/test/event/lifecycle/client/ClientTickTests.java
@@ -0,0 +1,56 @@
+/*
+ * 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.test.event.lifecycle.client;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import net.minecraft.util.registry.RegistryKey;
+import net.minecraft.world.World;
+
+import net.fabricmc.api.ClientModInitializer;
+import net.fabricmc.api.EnvType;
+import net.fabricmc.api.Environment;
+import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientTickEvents;
+import net.fabricmc.fabric.test.event.lifecycle.ServerLifecycleTests;
+
+@Environment(EnvType.CLIENT)
+public class ClientTickTests implements ClientModInitializer {
+	private Map<RegistryKey<World>, Integer> tickTracker = new HashMap<>();
+	private int ticks;
+
+	@Override
+	public void onInitializeClient() {
+		ClientTickEvents.END_CLIENT_TICK.register(client -> {
+			this.ticks++; // Just track our own tick since the client doesn't have a ticks value.
+
+			if (this.ticks % 200 == 0) {
+				ServerLifecycleTests.LOGGER.info("Ticked Client at " + this.ticks + " ticks.");
+			}
+		});
+
+		ClientTickEvents.END_WORLD_TICK.register(world -> {
+			final int worldTicks = this.tickTracker.computeIfAbsent(world.getRegistryKey(), k -> 0);
+
+			if (worldTicks % 200 == 0) { // Log every 200 ticks to verify the tick callback works on the client world
+				ServerLifecycleTests.LOGGER.info("Ticked Client World - " + worldTicks + " ticks:" + world.getRegistryKey());
+			}
+
+			this.tickTracker.put(world.getRegistryKey(), worldTicks + 1);
+		});
+	}
+}
diff --git a/fabric-lifecycle-events-v1/src/testmod/resources/fabric.mod.json b/fabric-lifecycle-events-v1/src/testmod/resources/fabric.mod.json
new file mode 100644
index 000000000..4ba0787c4
--- /dev/null
+++ b/fabric-lifecycle-events-v1/src/testmod/resources/fabric.mod.json
@@ -0,0 +1,25 @@
+{
+  "schemaVersion": 1,
+  "id": "fabric-lifecycle-events-v1-testmod",
+  "name": "Fabric Lifecycle Events (v1) Test Mod",
+  "version": "1.0.0",
+  "environment": "*",
+  "license": "Apache-2.0",
+  "depends": {
+    "fabric-lifecycle-events-v1": "*"
+  },
+  "entrypoints": {
+    "main": [
+      "net.fabricmc.fabric.test.event.lifecycle.ServerBlockEntityLifecycleTests",
+      "net.fabricmc.fabric.test.event.lifecycle.ServerEntityLifecycleTests",
+      "net.fabricmc.fabric.test.event.lifecycle.ServerLifecycleTests",
+      "net.fabricmc.fabric.test.event.lifecycle.ServerTickTests"
+    ],
+    "client": [
+      "net.fabricmc.fabric.test.event.lifecycle.client.ClientBlockEntityLifecycleTests",
+      "net.fabricmc.fabric.test.event.lifecycle.client.ClientEntityLifecycleTests",
+      "net.fabricmc.fabric.test.event.lifecycle.client.ClientLifecycleTests",
+      "net.fabricmc.fabric.test.event.lifecycle.client.ClientTickTests",
+    ]
+  }
+}
diff --git a/settings.gradle b/settings.gradle
index 4dcd34a31..3b458247c 100644
--- a/settings.gradle
+++ b/settings.gradle
@@ -24,9 +24,11 @@ include 'fabric-crash-report-info-v1'
 include 'fabric-dimensions-v1'
 include 'fabric-events-interaction-v0'
 include 'fabric-events-lifecycle-v0'
+include 'fabric-item-api-v1'
 include 'fabric-item-groups-v0'
 include 'fabric-keybindings-v0'
 include 'fabric-key-binding-api-v1'
+include 'fabric-lifecycle-events-v1'
 include 'fabric-loot-tables-v1'
 include 'fabric-mining-levels-v0'
 include 'fabric-models-v0'