diff --git a/assets/accretion_disk.svg b/assets/accretion_disk.svg
new file mode 100644
index 00000000..ee302045
--- /dev/null
+++ b/assets/accretion_disk.svg
@@ -0,0 +1,80 @@
+
+
+
+
diff --git a/assets/models/bulb_angry.png b/assets/models/bulb_angry.png
new file mode 100644
index 00000000..5f5cc603
Binary files /dev/null and b/assets/models/bulb_angry.png differ
diff --git a/assets/models/bulb_idle.png b/assets/models/bulb_idle.png
new file mode 100644
index 00000000..7fe663ff
Binary files /dev/null and b/assets/models/bulb_idle.png differ
diff --git a/assets/models/cloud_bed.java b/assets/models/cloud_bed.java
deleted file mode 100644
index 88008846..00000000
--- a/assets/models/cloud_bed.java
+++ /dev/null
@@ -1,50 +0,0 @@
-// Made with Blockbench 4.8.3
-// Exported for Minecraft version 1.17 or later with Mojang mappings
-// Paste this class into your mod and generate all required imports
-
-
-public class cloud_bed extends EntityModel {
- // This layer location should be baked with EntityRendererProvider.Context in the entity renderer and passed into this model's constructor
- public static final ModelLayerLocation LAYER_LOCATION = new ModelLayerLocation(new ResourceLocation("modid", "cloud_bed"), "main");
- private final ModelPart head;
- private final ModelPart foot;
-
- public cloud_bed(ModelPart root) {
- this.head = root.getChild("head");
- this.foot = root.getChild("foot");
- }
-
- public static LayerDefinition createBodyLayer() {
- MeshDefinition meshdefinition = new MeshDefinition();
- PartDefinition partdefinition = meshdefinition.getRoot();
-
- PartDefinition head = partdefinition.addOrReplaceChild("head", CubeListBuilder.create(), PartPose.offset(0.0F, 24.0F, 0.0F));
-
- PartDefinition main_r1 = head.addOrReplaceChild("main_r1", CubeListBuilder.create().texOffs(0, 0).addBox(-8.0F, -21.0F, -9.0F, 16.0F, 13.0F, 6.0F, new CubeDeformation(0.0F)), PartPose.offsetAndRotation(0.0F, 0.0F, -16.0F, -1.5708F, 0.0F, 0.0F));
-
- PartDefinition head_board = head.addOrReplaceChild("head_board", CubeListBuilder.create().texOffs(52, 24).addBox(7.0F, -13.0F, -3.0F, 2.0F, 13.0F, 3.0F, new CubeDeformation(0.0F))
- .texOffs(0, 43).addBox(-6.0F, -15.0F, -2.0F, 13.0F, 15.0F, 3.0F, new CubeDeformation(0.0F))
- .texOffs(52, 24).addBox(-8.0F, -13.0F, -3.0F, 2.0F, 13.0F, 3.0F, new CubeDeformation(0.0F)), PartPose.offset(-0.5F, 0.0F, 7.0F));
-
- PartDefinition foot = partdefinition.addOrReplaceChild("foot", CubeListBuilder.create(), PartPose.offset(0.0F, 24.0F, 0.0F));
-
- PartDefinition main_r2 = foot.addOrReplaceChild("main_r2", CubeListBuilder.create().texOffs(0, 22).addBox(-8.0F, -8.0F, -9.0F, 16.0F, 15.0F, 6.0F, new CubeDeformation(0.0F)), PartPose.offsetAndRotation(0.0F, 0.0F, 0.0F, -1.5708F, 0.0F, 0.0F));
-
- PartDefinition head_board2 = foot.addOrReplaceChild("head_board2", CubeListBuilder.create().texOffs(52, 40).addBox(6.0F, -13.0F, -2.0F, 3.0F, 10.0F, 3.0F, new CubeDeformation(0.0F))
- .texOffs(0, 43).addBox(-6.0F, -15.0F, -3.0F, 13.0F, 11.0F, 3.0F, new CubeDeformation(0.0F))
- .texOffs(52, 40).addBox(-8.0F, -13.0F, -2.0F, 3.0F, 10.0F, 3.0F, new CubeDeformation(0.0F)), PartPose.offset(-0.5F, 3.0F, -7.0F));
-
- return LayerDefinition.create(meshdefinition, 64, 64);
- }
-
- @Override
- public void setupAnim(T entity, float limbSwing, float limbSwingAmount, float ageInTicks, float netHeadYaw, float headPitch) {
-
- }
-
- @Override
- public void renderToBuffer(PoseStack poseStack, VertexConsumer vertexConsumer, int packedLight, int packedOverlay, float red, float green, float blue, float alpha) {
- head.render(poseStack, vertexConsumer, packedLight, packedOverlay, red, green, blue, alpha);
- foot.render(poseStack, vertexConsumer, packedLight, packedOverlay, red, green, blue, alpha);
- }
-}
\ No newline at end of file
diff --git a/assets/models/cloud_chest.java b/assets/models/cloud_chest.java
deleted file mode 100644
index 87e684a2..00000000
--- a/assets/models/cloud_chest.java
+++ /dev/null
@@ -1,32 +0,0 @@
-// Made with Blockbench 4.8.3
-// Exported for Minecraft version 1.17+ for Yarn
-// Paste this class into your mod and generate all required imports
-public class cloud_chest extends EntityModel {
- private final ModelPart lid;
- private final ModelPart lock_r1;
- private final ModelPart bb_main;
- public cloud_chest(ModelPart root) {
- this.lid = root.getChild("lid");
- this.bb_main = root.getChild("bb_main");
- }
- public static TexturedModelData getTexturedModelData() {
- ModelData modelData = new ModelData();
- ModelPartData modelPartData = modelData.getRoot();
- ModelPartData lid = modelPartData.addChild("lid", ModelPartBuilder.create().uv(0, 0).cuboid(-1.0F, -2.0F, 13.8F, 2.0F, 4.0F, 1.0F, new Dilation(0.0F))
- .uv(0, 0).cuboid(-1.0F, -1.0F, 14.0F, 2.0F, 2.0F, 1.0F, new Dilation(0.0F))
- .uv(0, 0).cuboid(-7.0F, -5.0F, 0.0F, 14.0F, 5.0F, 14.0F, new Dilation(0.3F)), ModelTransform.of(0.0F, 16.0F, -7.0F, 1.0908F, 0.0F, 0.0F));
-
- ModelPartData lock_r1 = lid.addChild("lock_r1", ModelPartBuilder.create().uv(0, 0).cuboid(-2.0F, -4.0F, -0.5F, 2.0F, 4.0F, 1.0F, new Dilation(0.0F)), ModelTransform.of(-2.0F, 1.0F, 14.3F, 0.0F, 0.0F, 1.5708F));
-
- ModelPartData bb_main = modelPartData.addChild("bb_main", ModelPartBuilder.create().uv(0, 19).cuboid(-7.0F, -10.0F, -7.0F, 14.0F, 10.0F, 14.0F, new Dilation(0.0F)), ModelTransform.pivot(0.0F, 24.0F, 0.0F));
- return TexturedModelData.of(modelData, 64, 64);
- }
- @Override
- public void setAngles(Entity entity, float limbSwing, float limbSwingAmount, float ageInTicks, float netHeadYaw, float headPitch) {
- }
- @Override
- public void render(MatrixStack matrices, VertexConsumer vertexConsumer, int light, int overlay, float red, float green, float blue, float alpha) {
- lid.render(matrices, vertexConsumer, light, overlay, red, green, blue, alpha);
- bb_main.render(matrices, vertexConsumer, light, overlay, red, green, blue, alpha);
- }
-}
\ No newline at end of file
diff --git a/assets/models/hay_bale.bbmodel b/assets/models/hay_bale.bbmodel
new file mode 100644
index 00000000..7328ccde
--- /dev/null
+++ b/assets/models/hay_bale.bbmodel
@@ -0,0 +1 @@
+{"meta":{"format_version":"4.5","model_format":"java_block","box_uv":false},"name":"hay_bale_bse","parent":"","ambientocclusion":true,"front_gui_light":false,"visible_box":[1,1,0],"variable_placeholders":"","variable_placeholder_buttons":[],"unhandled_root_fields":{},"resolution":{"width":16,"height":16},"elements":[{"name":"bottom_south_east","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[8,0,8],"to":[16,8,16],"autouv":0,"color":2,"origin":[0,0,0],"faces":{"north":{"uv":[0,8,8,16],"texture":0},"east":{"uv":[0,8,8,16],"texture":1},"south":{"uv":[8,8,16,16],"texture":1},"west":{"uv":[8,8,16,16],"texture":0},"up":{"uv":[8,8,16,16],"texture":0},"down":{"uv":[8,0,16,8],"texture":0}},"type":"cube","uuid":"523f4164-daa5-4b79-eec7-013710b7ae88"},{"name":"bottom_north_west","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[0,0,0],"to":[8,8,8],"autouv":0,"color":2,"origin":[0,0,0],"faces":{"north":{"uv":[8,8,16,16],"texture":1},"east":{"uv":[8,8,16,16],"texture":0},"south":{"uv":[0,8,8,16],"texture":0},"west":{"uv":[0,8,8,16],"texture":1},"up":{"uv":[0,0,8,8],"texture":0},"down":{"uv":[0,8,8,16],"texture":0}},"type":"cube","uuid":"1fd288fe-a293-1ca2-a5b7-15d602eb44f6"},{"name":"top_south_east","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[8,8,8],"to":[16,16,16],"autouv":0,"color":2,"origin":[0,0,0],"faces":{"north":{"uv":[0,0,8,8],"texture":0},"east":{"uv":[0,0,8,8],"texture":1},"south":{"uv":[8,0,16,8],"texture":1},"west":{"uv":[8,0,16,8],"texture":0},"up":{"uv":[8,8,16,16],"texture":0},"down":{"uv":[8,0,16,8],"texture":0}},"type":"cube","uuid":"4f48967d-c722-5593-6f77-079ee47d08af"},{"name":"top_north_west","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[0,8,0],"to":[8,16,8],"autouv":0,"color":2,"origin":[0,0,0],"faces":{"north":{"uv":[8,0,16,8],"texture":1},"east":{"uv":[8,0,16,8],"texture":0},"south":{"uv":[0,0,8,8],"texture":0},"west":{"uv":[0,0,8,8],"texture":1},"up":{"uv":[0,0,8,8],"texture":0},"down":{"uv":[0,8,8,16],"texture":0}},"type":"cube","uuid":"c4356ab3-a7a5-43e5-489a-9ebb52492675"},{"name":"bottom_south_west","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[0,0,8],"to":[8,8,16],"autouv":0,"color":2,"origin":[0,0,0],"faces":{"north":{"uv":[8,8,16,16],"texture":0},"east":{"uv":[0,8,8,16],"texture":0},"south":{"uv":[0,8,8,16],"texture":1},"west":{"uv":[8,8,16,16],"texture":1},"up":{"uv":[0,8,8,16],"texture":0},"down":{"uv":[0,0,8,8],"texture":0}},"type":"cube","uuid":"d07d02d7-5ab9-0026-b3ed-029701536cf6"},{"name":"bottom_north_east","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[8,0,0],"to":[16,8,8],"autouv":0,"color":2,"origin":[0,0,0],"faces":{"north":{"uv":[0,8,8,16],"texture":1},"east":{"uv":[8,8,16,16],"texture":1},"south":{"uv":[8,8,16,16],"texture":0},"west":{"uv":[0,8,8,16],"texture":0},"up":{"uv":[8,0,16,8],"texture":0},"down":{"uv":[8,8,16,16],"texture":0}},"type":"cube","uuid":"d10fb8d4-fe44-ca10-77f0-63b32855f5d6"},{"name":"top_south_west","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[0,8,8],"to":[8,16,16],"autouv":0,"color":2,"origin":[0,0,0],"faces":{"north":{"uv":[8,0,16,8],"texture":0},"east":{"uv":[0,0,8,8],"texture":0},"south":{"uv":[0,0,8,8],"texture":1},"west":{"uv":[8,0,16,8],"texture":1},"up":{"uv":[0,8,8,16],"texture":0},"down":{"uv":[0,0,8,8],"texture":0}},"type":"cube","uuid":"b9c9d695-4003-c9e6-efbb-9a75c658b47e"},{"name":"top_north_east","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[8,8,0],"to":[16,16,8],"autouv":0,"color":2,"origin":[0,0,0],"faces":{"north":{"uv":[0,0,8,8],"texture":1},"east":{"uv":[8,0,16,8],"texture":1},"south":{"uv":[8,0,16,8],"texture":0},"west":{"uv":[0,0,8,8],"texture":0},"up":{"uv":[8,0,16,8],"texture":0},"down":{"uv":[8,8,16,16],"texture":0}},"type":"cube","uuid":"20a14bb3-edf7-1ae1-132e-ef3d56d84f67"}],"outliner":["523f4164-daa5-4b79-eec7-013710b7ae88","4f48967d-c722-5593-6f77-079ee47d08af","d10fb8d4-fe44-ca10-77f0-63b32855f5d6","20a14bb3-edf7-1ae1-132e-ef3d56d84f67","d07d02d7-5ab9-0026-b3ed-029701536cf6","b9c9d695-4003-c9e6-efbb-9a75c658b47e","1fd288fe-a293-1ca2-a5b7-15d602eb44f6","c4356ab3-a7a5-43e5-489a-9ebb52492675"],"textures":[{"path":"/home/sollace/Desktop/hay_block_top.png","name":"hay_block_top.png","folder":"blocks","namespace":"minecraft","id":"top","particle":false,"render_mode":"default","render_sides":"auto","frame_time":1,"frame_order_type":"loop","frame_order":"","frame_interpolate":false,"visible":true,"mode":"bitmap","saved":true,"uuid":"f73f23e2-d591-dd05-db6f-60869e1592c7","relative_path":"../../../../../../../Desktop/hay_block_top.png","source":""},{"path":"/home/sollace/Desktop/hay_block_side.png","name":"hay_block_side.png","folder":"blocks","namespace":"minecraft","id":"side","particle":true,"render_mode":"default","render_sides":"auto","frame_time":1,"frame_order_type":"loop","frame_order":"","frame_interpolate":false,"visible":true,"mode":"bitmap","saved":true,"uuid":"1cf36f16-9f26-3125-d49c-45217088cbc7","relative_path":"../../../../../../../Desktop/hay_block_side.png","source":""}]}
\ No newline at end of file
diff --git a/assets/models/hay_bale.json b/assets/models/hay_bale.json
new file mode 100644
index 00000000..a2ce1538
--- /dev/null
+++ b/assets/models/hay_bale.json
@@ -0,0 +1,113 @@
+{
+ "textures": {
+ "top": "blocks/hay_block_top",
+ "particle": "blocks/hay_block_side",
+ "side": "blocks/hay_block_side"
+ },
+ "elements": [
+ {
+ "name": "bottom_south_east",
+ "from": [8, 0, 8],
+ "to": [16, 8, 16],
+ "faces": {
+ "north": {"uv": [0, 8, 8, 16], "texture": "#top"},
+ "east": {"uv": [0, 8, 8, 16], "texture": "#side"},
+ "south": {"uv": [8, 8, 16, 16], "texture": "#side"},
+ "west": {"uv": [8, 8, 16, 16], "texture": "#top"},
+ "up": {"uv": [8, 8, 16, 16], "texture": "#top"},
+ "down": {"uv": [8, 0, 16, 8], "texture": "#top"}
+ }
+ },
+ {
+ "name": "top_south_east",
+ "from": [8, 8, 8],
+ "to": [16, 16, 16],
+ "faces": {
+ "north": {"uv": [0, 0, 8, 8], "texture": "#top"},
+ "east": {"uv": [0, 0, 8, 8], "texture": "#side"},
+ "south": {"uv": [8, 0, 16, 8], "texture": "#side"},
+ "west": {"uv": [8, 0, 16, 8], "texture": "#top"},
+ "up": {"uv": [8, 8, 16, 16], "texture": "#top"},
+ "down": {"uv": [8, 0, 16, 8], "texture": "#top"}
+ }
+ },
+ {
+ "name": "bottom_north_east",
+ "from": [8, 0, 0],
+ "to": [16, 8, 8],
+ "faces": {
+ "north": {"uv": [0, 8, 8, 16], "texture": "#side"},
+ "east": {"uv": [8, 8, 16, 16], "texture": "#side"},
+ "south": {"uv": [8, 8, 16, 16], "texture": "#top"},
+ "west": {"uv": [0, 8, 8, 16], "texture": "#top"},
+ "up": {"uv": [8, 0, 16, 8], "texture": "#top"},
+ "down": {"uv": [8, 8, 16, 16], "texture": "#top"}
+ }
+ },
+ {
+ "name": "top_north_east",
+ "from": [8, 8, 0],
+ "to": [16, 16, 8],
+ "faces": {
+ "north": {"uv": [0, 0, 8, 8], "texture": "#side"},
+ "east": {"uv": [8, 0, 16, 8], "texture": "#side"},
+ "south": {"uv": [8, 0, 16, 8], "texture": "#top"},
+ "west": {"uv": [0, 0, 8, 8], "texture": "#top"},
+ "up": {"uv": [8, 0, 16, 8], "texture": "#top"},
+ "down": {"uv": [8, 8, 16, 16], "texture": "#top"}
+ }
+ },
+ {
+ "name": "bottom_south_west",
+ "from": [0, 0, 8],
+ "to": [8, 8, 16],
+ "faces": {
+ "north": {"uv": [8, 8, 16, 16], "texture": "#top"},
+ "east": {"uv": [0, 8, 8, 16], "texture": "#top"},
+ "south": {"uv": [0, 8, 8, 16], "texture": "#side"},
+ "west": {"uv": [8, 8, 16, 16], "texture": "#side"},
+ "up": {"uv": [0, 8, 8, 16], "texture": "#top"},
+ "down": {"uv": [0, 0, 8, 8], "texture": "#top"}
+ }
+ },
+ {
+ "name": "top_south_west",
+ "from": [0, 8, 8],
+ "to": [8, 16, 16],
+ "faces": {
+ "north": {"uv": [8, 0, 16, 8], "texture": "#top"},
+ "east": {"uv": [0, 0, 8, 8], "texture": "#top"},
+ "south": {"uv": [0, 0, 8, 8], "texture": "#side"},
+ "west": {"uv": [8, 0, 16, 8], "texture": "#side"},
+ "up": {"uv": [0, 8, 8, 16], "texture": "#top"},
+ "down": {"uv": [0, 0, 8, 8], "texture": "#top"}
+ }
+ },
+ {
+ "name": "bottom_north_west",
+ "from": [0, 0, 0],
+ "to": [8, 8, 8],
+ "faces": {
+ "north": {"uv": [8, 8, 16, 16], "texture": "#side"},
+ "east": {"uv": [8, 8, 16, 16], "texture": "#top"},
+ "south": {"uv": [0, 8, 8, 16], "texture": "#top"},
+ "west": {"uv": [0, 8, 8, 16], "texture": "#side"},
+ "up": {"uv": [0, 0, 8, 8], "texture": "#top"},
+ "down": {"uv": [0, 8, 8, 16], "texture": "#top"}
+ }
+ },
+ {
+ "name": "top_north_west",
+ "from": [0, 8, 0],
+ "to": [8, 16, 8],
+ "faces": {
+ "north": {"uv": [8, 0, 16, 8], "texture": "#side"},
+ "east": {"uv": [8, 0, 16, 8], "texture": "#top"},
+ "south": {"uv": [0, 0, 8, 8], "texture": "#top"},
+ "west": {"uv": [0, 0, 8, 8], "texture": "#side"},
+ "up": {"uv": [0, 0, 8, 8], "texture": "#top"},
+ "down": {"uv": [0, 8, 8, 16], "texture": "#top"}
+ }
+ }
+ ]
+}
\ No newline at end of file
diff --git a/assets/models/ignimious_bulb.bbmodel b/assets/models/ignimious_bulb.bbmodel
new file mode 100644
index 00000000..2c3b6593
--- /dev/null
+++ b/assets/models/ignimious_bulb.bbmodel
@@ -0,0 +1 @@
+{"meta":{"format_version":"4.5","model_format":"modded_entity","box_uv":true},"name":"ignimious_bulb","model_identifier":"","modded_entity_version":"Fabric 1.17+","modded_entity_flip_y":true,"visible_box":[1,1,0],"variable_placeholders":"","variable_placeholder_buttons":[],"unhandled_root_fields":{},"resolution":{"width":256,"height":256},"elements":[{"name":"jaw","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-23.956753782540364,-5.009498774448022,-24.13052619222006],"to":[24.043246217459636,17.990501225551977,23.86947380777994],"autouv":0,"color":8,"origin":[0.04324621745963608,-5.009498774448022,-0.13052619222006],"faces":{"north":{"uv":[48,48,96,71],"texture":0},"east":{"uv":[0,48,48,71],"texture":0},"south":{"uv":[144,48,192,71],"texture":0},"west":{"uv":[96,48,144,71],"texture":0},"up":{"uv":[96,48,48,0],"texture":0},"down":{"uv":[144,0,96,48],"texture":0}},"type":"cube","uuid":"800563f2-e637-d42d-6c38-616b6b34c9ca"},{"name":"head","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-22.956753782540364,10.990501225551977,-22.13052619222006],"to":[23.043246217459636,33.99050122555197,23.86947380777994],"autouv":0,"color":8,"rotation":[7.5,0,0],"origin":[0.04324621745963608,17.990501225551977,23.86947380777994],"uv_offset":[0,71],"faces":{"north":{"uv":[46,117,92,140],"texture":0},"east":{"uv":[0,117,46,140],"texture":0},"south":{"uv":[138,117,184,140],"texture":0},"west":{"uv":[92,117,138,140],"texture":0},"up":{"uv":[92,117,46,71],"texture":0},"down":{"uv":[138,71,92,117],"texture":0}},"type":"cube","uuid":"e6d05459-6b50-12b5-a46a-eb72b97f7c09"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-19,-9,-61],"to":[13,-9,-29],"autouv":0,"color":9,"rotation":[27.320650966819105,-50.33976442181192,-6.793186828559589],"origin":[0,-3,-1],"uv_offset":[112,0],"faces":{"north":{"uv":[144,32,176,32],"texture":0},"east":{"uv":[112,32,144,32],"texture":0},"south":{"uv":[208,32,240,32],"texture":0},"west":{"uv":[176,32,208,32],"texture":0},"up":{"uv":[176,32,144,0],"texture":0},"down":{"uv":[208,0,176,32],"texture":0}},"type":"cube","uuid":"42f404db-2eb0-2efd-bb09-8810376fcfbc"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-16,-3,-60],"to":[16,-3,-28],"autouv":0,"color":9,"rotation":[16.681567243909033,33.56061965535128,3.4685333161320835],"origin":[0,-3,-1],"uv_offset":[112,0],"faces":{"north":{"uv":[144,32,176,32],"texture":0},"east":{"uv":[112,32,144,32],"texture":0},"south":{"uv":[208,32,240,32],"texture":0},"west":{"uv":[176,32,208,32],"texture":0},"up":{"uv":[176,32,144,0],"texture":0},"down":{"uv":[208,0,176,32],"texture":0}},"type":"cube","uuid":"4c00dc9a-fef0-30b4-0f06-8bee9fe9b898"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-16,-6,-56],"to":[16,-6,-24],"autouv":0,"color":9,"rotation":[-162.2403464416216,58.02175378934316,-176.80418113816847],"origin":[0,-3,-1],"uv_offset":[112,0],"faces":{"north":{"uv":[144,32,176,32],"texture":0},"east":{"uv":[112,32,144,32],"texture":0},"south":{"uv":[208,32,240,32],"texture":0},"west":{"uv":[176,32,208,32],"texture":0},"up":{"uv":[176,32,144,0],"texture":0},"down":{"uv":[208,0,176,32],"texture":0}},"type":"cube","uuid":"5db8a4ea-e690-f657-b64a-323ec0c3a305"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-24,-6,-62],"to":[8,-6,-30],"autouv":0,"color":9,"rotation":[-169.75684992659043,-35.13964450125722,-170.6241287065347],"origin":[0,-3,-1],"uv_offset":[112,0],"faces":{"north":{"uv":[144,32,176,32],"texture":0},"east":{"uv":[112,32,144,32],"texture":0},"south":{"uv":[208,32,240,32],"texture":0},"west":{"uv":[176,32,208,32],"texture":0},"up":{"uv":[176,32,144,0],"texture":0},"down":{"uv":[208,0,176,32],"texture":0}},"type":"cube","uuid":"b2bbe86a-3379-0246-9181-127aed715432"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-16,-13,-58],"to":[16,-13,-26],"autouv":0,"color":9,"rotation":[144.36680378552091,-83.99459398899594,-115.75078983303227],"origin":[0,-3,-1],"uv_offset":[112,0],"faces":{"north":{"uv":[144,32,176,32],"texture":0},"east":{"uv":[112,32,144,32],"texture":0},"south":{"uv":[208,32,240,32],"texture":0},"west":{"uv":[176,32,208,32],"texture":0},"up":{"uv":[176,32,144,0],"texture":0},"down":{"uv":[208,0,176,32],"texture":0}},"type":"cube","uuid":"c63affed-9e2a-84af-5766-631a13679757"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-16,-9,-59],"to":[16,-9,-27],"autouv":0,"color":9,"rotation":[28.459049469615778,-5.7440354970636935,0.03610759351412133],"origin":[0,-3,-1],"uv_offset":[112,0],"faces":{"north":{"uv":[144,32,176,32],"texture":0},"east":{"uv":[112,32,144,32],"texture":0},"south":{"uv":[208,32,240,32],"texture":0},"west":{"uv":[176,32,208,32],"texture":0},"up":{"uv":[176,32,144,0],"texture":0},"down":{"uv":[208,0,176,32],"texture":0}},"type":"cube","uuid":"89bc35ae-6e61-dfa8-745f-0e317bb63260"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-12,-4,-52],"to":[20,-4,-20],"autouv":0,"color":9,"rotation":[40.560964930961916,83.77970397136126,27.74070945144022],"origin":[0,-3,-1],"uv_offset":[112,0],"faces":{"north":{"uv":[144,32,176,32],"texture":0},"east":{"uv":[112,32,144,32],"texture":0},"south":{"uv":[208,32,240,32],"texture":0},"west":{"uv":[176,32,208,32],"texture":0},"up":{"uv":[176,32,144,0],"texture":0},"down":{"uv":[208,0,176,32],"texture":0}},"type":"cube","uuid":"0a5c37c6-ad13-697a-58cd-934841edb237"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-18,-15,-61],"to":[14,-15,-29],"autouv":0,"color":9,"rotation":[-149.33066739822777,4.299459188610997,-178.742402181358],"origin":[0,-3,-1],"uv_offset":[112,0],"faces":{"north":{"uv":[144,32,176,32],"texture":0},"east":{"uv":[112,32,144,32],"texture":0},"south":{"uv":[208,32,240,32],"texture":0},"west":{"uv":[176,32,208,32],"texture":0},"up":{"uv":[176,32,144,0],"texture":0},"down":{"uv":[208,0,176,32],"texture":0}},"type":"cube","uuid":"d8b4bbe0-9707-0652-35e6-e28a54179fa3"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-34,-10,-84],"to":[30,-10,-20],"autouv":0,"color":9,"rotation":[111.7261647239295,-85.5550612830785,-96.58324240444844],"origin":[0,-3,-1],"uv_offset":[-64,140],"faces":{"north":{"uv":[0,204,64,204],"texture":0},"east":{"uv":[-64,204,0,204],"texture":0},"south":{"uv":[128,204,192,204],"texture":0},"west":{"uv":[64,204,128,204],"texture":0},"up":{"uv":[64,204,0,140],"texture":0},"down":{"uv":[128,140,64,204],"texture":0}},"type":"cube","uuid":"4e7c3e97-535b-9261-83a4-80fb76cc13d4"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-34,-10,-84],"to":[30,-10,-20],"autouv":0,"color":9,"rotation":[62.10613287479732,85.7200584727182,46.23842221938251],"origin":[0,-3,-1],"uv_offset":[-64,140],"faces":{"north":{"uv":[0,204,64,204],"texture":0},"east":{"uv":[-64,204,0,204],"texture":0},"south":{"uv":[128,204,192,204],"texture":0},"west":{"uv":[64,204,128,204],"texture":0},"up":{"uv":[64,204,0,140],"texture":0},"down":{"uv":[128,140,64,204],"texture":0}},"type":"cube","uuid":"2184d35d-e78b-c493-b1c4-a817cb45fa8b"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-34,-10,-84],"to":[30,-10,-20],"autouv":0,"color":9,"rotation":[-164.484342402648,7.951483564968085,-175.61972440808503],"origin":[0,-3,-1],"uv_offset":[-64,140],"faces":{"north":{"uv":[0,204,64,204],"texture":0},"east":{"uv":[-64,204,0,204],"texture":0},"south":{"uv":[128,204,192,204],"texture":0},"west":{"uv":[64,204,128,204],"texture":0},"up":{"uv":[64,204,0,140],"texture":0},"down":{"uv":[128,140,64,204],"texture":0}},"type":"cube","uuid":"24212edf-8ab0-e1e5-b67e-dd27ca7fee4f"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-34,-6,-84],"to":[30,-6,-20],"autouv":0,"color":9,"rotation":[15.186954129789529,2.0049709481247944,-1.9011648291348386],"origin":[0,-3,-1],"uv_offset":[-64,140],"faces":{"north":{"uv":[0,204,64,204],"texture":0},"east":{"uv":[-64,204,0,204],"texture":0},"south":{"uv":[128,204,192,204],"texture":0},"west":{"uv":[64,204,128,204],"texture":0},"up":{"uv":[64,204,0,140],"texture":0},"down":{"uv":[128,140,64,204],"texture":0}},"type":"cube","uuid":"dfd15693-f566-9bc7-b953-d303dec3ba18"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-17.373117897112767,65.61517788582364,17.389978221486494],"to":[14.626882102887233,65.61517788582364,49.389978221486494],"autouv":0,"color":9,"rotation":[113.4590494696158,-5.744035497063647,0.03610759351412453],"origin":[-1.3731178971127687,38.61517788582365,12.389978221486501],"uv_offset":[112,0],"faces":{"north":{"uv":[144,32,176,32],"texture":0},"east":{"uv":[112,32,144,32],"texture":0},"south":{"uv":[208,32,240,32],"texture":0},"west":{"uv":[176,32,208,32],"texture":0},"up":{"uv":[176,32,144,0],"texture":0},"down":{"uv":[208,0,176,32],"texture":0}},"type":"cube","uuid":"e020ca0b-24eb-a3cc-43ed-e87b0013e04d"}],"outliner":[{"name":"head","origin":[0,17,24],"rotation":[7.492942810808386,-0.32621370972236063,2.478638942680466],"color":0,"uuid":"e54419b9-7a64-8b11-6ad9-cea74cb14f41","export":true,"mirror_uv":false,"isOpen":true,"locked":false,"visibility":true,"autouv":0,"children":["e6d05459-6b50-12b5-a46a-eb72b97f7c09","800563f2-e637-d42d-6c38-616b6b34c9ca","e020ca0b-24eb-a3cc-43ed-e87b0013e04d"]},{"name":"leaves","origin":[0,-3,-1],"color":0,"uuid":"7cf17cab-353d-fddc-814d-3b708cb33637","export":true,"mirror_uv":false,"isOpen":true,"locked":false,"visibility":true,"autouv":0,"children":["4c00dc9a-fef0-30b4-0f06-8bee9fe9b898","42f404db-2eb0-2efd-bb09-8810376fcfbc","b2bbe86a-3379-0246-9181-127aed715432","c63affed-9e2a-84af-5766-631a13679757","4e7c3e97-535b-9261-83a4-80fb76cc13d4","dfd15693-f566-9bc7-b953-d303dec3ba18","2184d35d-e78b-c493-b1c4-a817cb45fa8b","24212edf-8ab0-e1e5-b67e-dd27ca7fee4f","89bc35ae-6e61-dfa8-745f-0e317bb63260","0a5c37c6-ad13-697a-58cd-934841edb237","d8b4bbe0-9707-0652-35e6-e28a54179fa3","5db8a4ea-e690-f657-b64a-323ec0c3a305"]}],"textures":[{"path":"/home/sollace/Documents/GitRepos/minecraft_mods/Unicopia/assets/models/bulb.png","name":"bulb.png","folder":"block","namespace":"","id":"0","particle":false,"render_mode":"default","render_sides":"auto","frame_time":1,"frame_order_type":"loop","frame_order":"","frame_interpolate":false,"visible":true,"mode":"bitmap","saved":true,"uuid":"742fdfc6-ad6a-1d7e-e84d-c8133537cff8","relative_path":"../bulb.png","source":""}],"fabricOptions":{"header":"package com.example.mod;","entity":"Entity","render":"","members":""}}
\ No newline at end of file
diff --git a/assets/models/slime_pustule_cap.json b/assets/models/slime_pustule_cap.json
deleted file mode 100644
index f337aa4f..00000000
--- a/assets/models/slime_pustule_cap.json
+++ /dev/null
@@ -1,46 +0,0 @@
-{
- "parent": "minecraft:block/cube_all",
- "textures": {
- "all": "unicopia:block/slime_pustule",
- "particle": "unicopia:block/slime_pustule"
- },
- "elements": [
- {
- "from": [7, 14.9, 7],
- "to": [12, 16, 12],
- "faces": {
- "north": {"uv": [0, 0, 6, 1.1], "texture": "#all"},
- "east": {"uv": [0, 0, 6, 1.1], "texture": "#all"},
- "south": {"uv": [0, 0, 6, 1.1], "texture": "#all"},
- "west": {"uv": [0, 0, 6, 1.1], "texture": "#all"},
- "up": {"uv": [0, 0, 6, 6], "texture": "#all"},
- "down": {"uv": [0, 0, 6, 6], "texture": "#all"}
- }
- },
- {
- "from": [3, 15, 4],
- "to": [9, 16, 10],
- "faces": {
- "north": {"uv": [0, 0, 6, 1], "texture": "#all"},
- "east": {"uv": [0, 0, 6, 1], "texture": "#all"},
- "south": {"uv": [0, 0, 6, 1], "texture": "#all"},
- "west": {"uv": [0, 0, 6, 1], "texture": "#all"},
- "up": {"uv": [0, 0, 6, 6], "texture": "#all"},
- "down": {"uv": [0, 0, 6, 6], "texture": "#all"}
- }
- },
- {
- "name": "rope",
- "from": [7, 10, 7],
- "to": [9, 15, 9],
- "faces": {
- "north": {"uv": [1, 7, 5, 13], "texture": "#all"},
- "east": {"uv": [1, 7, 5, 13], "texture": "#all"},
- "south": {"uv": [1, 7, 5, 13], "texture": "#all"},
- "west": {"uv": [1, 7, 5, 13], "texture": "#all"},
- "up": {"uv": [2, 2, 4, 4], "texture": "#all"},
- "down": {"uv": [2, 2, 4, 4], "texture": "#all"}
- }
- }
- ]
-}
\ No newline at end of file
diff --git a/assets/models/slime_pustule_pod.json b/assets/models/slime_pustule_pod.json
deleted file mode 100644
index 7c15dd7b..00000000
--- a/assets/models/slime_pustule_pod.json
+++ /dev/null
@@ -1,137 +0,0 @@
-{
- "parent": "minecraft:block/cube_all",
- "textures": {
- "all": "unicopia:block/slime_pustule",
- "particle": "unicopia:block/slime_pustule"
- },
- "elements": [
- {
- "name": "rope",
- "from": [7.5, 0, 7.5],
- "to": [8.5, 16, 8.5],
- "faces": {
- "north": {"uv": [12, 0, 13, 16], "texture": "#all"},
- "east": {"uv": [13, 0, 14, 16], "texture": "#all"},
- "south": {"uv": [15, 0, 16, 16], "texture": "#all"},
- "west": {"uv": [14, 0, 15, 16], "texture": "#all"},
- "up": {"uv": [13, 0, 14, 1], "texture": "#all"},
- "down": {"uv": [12, 0, 13, 1], "texture": "#all"}
- }
- },
- {
- "from": [7, 14.9, 7],
- "to": [12, 16, 12],
- "faces": {
- "north": {"uv": [0, 0, 6, 1.1], "texture": "#all"},
- "east": {"uv": [0, 0, 6, 1.1], "texture": "#all"},
- "south": {"uv": [0, 0, 6, 1.1], "texture": "#all"},
- "west": {"uv": [0, 0, 6, 1.1], "texture": "#all"},
- "up": {"uv": [0, 0, 6, 6], "texture": "#all"},
- "down": {"uv": [0, 0, 6, 6], "texture": "#all"}
- }
- },
- {
- "from": [3, 15, 4],
- "to": [9, 16, 10],
- "faces": {
- "north": {"uv": [0, 0, 6, 1], "texture": "#all"},
- "east": {"uv": [0, 0, 6, 1], "texture": "#all"},
- "south": {"uv": [0, 0, 6, 1], "texture": "#all"},
- "west": {"uv": [0, 0, 6, 1], "texture": "#all"},
- "up": {"uv": [0, 0, 6, 6], "texture": "#all"},
- "down": {"uv": [0, 0, 6, 6], "texture": "#all"}
- }
- },
- {
- "name": "rope",
- "from": [7, 10, 7],
- "to": [9, 15, 9],
- "faces": {
- "north": {"uv": [1, 7, 5, 13], "texture": "#all"},
- "east": {"uv": [1, 7, 5, 13], "texture": "#all"},
- "south": {"uv": [1, 7, 5, 13], "texture": "#all"},
- "west": {"uv": [1, 7, 5, 13], "texture": "#all"},
- "up": {"uv": [2, 2, 4, 4], "texture": "#all"},
- "down": {"uv": [2, 2, 4, 4], "texture": "#all"}
- }
- },
- {
- "name": "rope",
- "from": [7, 15, 7],
- "to": [9, 20, 9],
- "faces": {
- "north": {"uv": [2, 7, 4, 13], "texture": "#all"},
- "east": {"uv": [2, 7, 4, 13], "texture": "#all"},
- "south": {"uv": [2, 7, 4, 13], "texture": "#all"},
- "west": {"uv": [2, 7, 4, 13], "texture": "#all"},
- "up": {"uv": [2, 2, 4, 4], "texture": "#all"},
- "down": {"uv": [2, 2, 4, 4], "texture": "#all"}
- }
- },
- {
- "name": "drop",
- "from": [5, 10, 5],
- "to": [11, 13, 11],
- "faces": {
- "north": {"uv": [0, 6, 6, 9], "texture": "#all"},
- "east": {"uv": [0, 6, 6, 9], "texture": "#all"},
- "south": {"uv": [0, 6, 6, 9], "texture": "#all"},
- "west": {"uv": [0, 6, 6, 9], "texture": "#all"},
- "up": {"uv": [0, 0, 6, 6], "texture": "#all"},
- "down": {"uv": [0, 0, 6, 6], "texture": "#all"}
- }
- },
- {
- "name": "drop",
- "from": [6, 13, 6],
- "to": [10, 15, 10],
- "faces": {
- "north": {"uv": [0, 6, 6, 8], "texture": "#all"},
- "east": {"uv": [0, 6, 6, 8], "texture": "#all"},
- "south": {"uv": [0, 6, 6, 8], "texture": "#all"},
- "west": {"uv": [0, 6, 6, 8], "texture": "#all"},
- "up": {"uv": [0, 0, 6, 6], "texture": "#all"},
- "down": {"uv": [0, 0, 6, 6], "texture": "#all"}
- }
- },
- {
- "name": "drop",
- "from": [5, 0, 5],
- "to": [11, 1, 11],
- "faces": {
- "north": {"uv": [0, 13, 6, 14], "texture": "#all"},
- "east": {"uv": [0, 13, 6, 14], "texture": "#all"},
- "south": {"uv": [0, 13, 6, 14], "texture": "#all"},
- "west": {"uv": [0, 13, 6, 14], "texture": "#all"},
- "up": {"uv": [0, 0, 6, 6], "texture": "#all"},
- "down": {"uv": [0, 0, 6, 6], "texture": "#all"}
- }
- },
- {
- "name": "drop",
- "from": [4, 1, 4],
- "to": [12, 10, 12],
- "faces": {
- "north": {"uv": [0, 6, 6, 14], "texture": "#all"},
- "east": {"uv": [0, 6, 6, 14], "texture": "#all"},
- "south": {"uv": [0, 6, 6, 14], "texture": "#all"},
- "west": {"uv": [0, 6, 6, 14], "texture": "#all"},
- "up": {"uv": [0, 0, 6, 6], "texture": "#all"},
- "down": {"uv": [0, 0, 6, 6], "texture": "#all"}
- }
- },
- {
- "name": "drop",
- "from": [5, 2, 5],
- "to": [11, 9, 11],
- "faces": {
- "north": {"uv": [7, 7, 11, 13], "texture": "#all"},
- "east": {"uv": [7, 7, 11, 13], "texture": "#all"},
- "south": {"uv": [7, 7, 11, 13], "texture": "#all"},
- "west": {"uv": [7, 7, 11, 13], "texture": "#all"},
- "up": {"uv": [7, 1, 11, 5], "texture": "#all"},
- "down": {"uv": [7, 1, 11, 5], "texture": "#all"}
- }
- }
- ]
-}
\ No newline at end of file
diff --git a/assets/models/slime_pustule_rope.json b/assets/models/slime_pustule_rope.json
deleted file mode 100644
index aab23dfc..00000000
--- a/assets/models/slime_pustule_rope.json
+++ /dev/null
@@ -1,22 +0,0 @@
-{
- "parent": "minecraft:block/cube_all",
- "textures": {
- "all": "unicopia:block/slime_pustule",
- "particle": "unicopia:block/slime_pustule"
- },
- "elements": [
- {
- "name": "rope",
- "from": [7.5, 0, 7.5],
- "to": [8.5, 16, 8.5],
- "faces": {
- "north": {"uv": [12, 0, 13, 16], "texture": "#all"},
- "east": {"uv": [13, 0, 14, 16], "texture": "#all"},
- "south": {"uv": [15, 0, 16, 16], "texture": "#all"},
- "west": {"uv": [14, 0, 15, 16], "texture": "#all"},
- "up": {"uv": [13, 0, 14, 1], "texture": "#all"},
- "down": {"uv": [12, 0, 13, 1], "texture": "#all"}
- }
- }
- ]
-}
\ No newline at end of file
diff --git a/assets/models/tentacle.bbmodel b/assets/models/tentacle.bbmodel
new file mode 100644
index 00000000..0ccec62c
--- /dev/null
+++ b/assets/models/tentacle.bbmodel
@@ -0,0 +1 @@
+{"meta":{"format_version":"4.5","model_format":"modded_entity","box_uv":true},"name":"tentacle","model_identifier":"","modded_entity_version":"Fabric 1.17+","modded_entity_flip_y":true,"visible_box":[1,1,0],"variable_placeholders":"","variable_placeholder_buttons":[],"unhandled_root_fields":{},"resolution":{"width":128,"height":128},"elements":[{"name":"a","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-7,-6,-7],"to":[7,10,7],"autouv":0,"color":7,"origin":[0,-6,0],"faces":{"north":{"uv":[14,14,28,30],"texture":0},"east":{"uv":[0,14,14,30],"texture":0},"south":{"uv":[42,14,56,30],"texture":0},"west":{"uv":[28,14,42,30],"texture":0},"up":{"uv":[28,14,14,0],"texture":0},"down":{"uv":[42,0,28,14],"texture":0}},"type":"cube","uuid":"43deca74-ef0e-9496-f060-a569a5f66a3c"},{"name":"b","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-6,8,-6],"to":[6,27,6],"autouv":0,"color":7,"origin":[0,0,0],"uv_offset":[0,30],"faces":{"north":{"uv":[12,42,24,61],"texture":0},"east":{"uv":[0,42,12,61],"texture":0},"south":{"uv":[36,42,48,61],"texture":0},"west":{"uv":[24,42,36,61],"texture":0},"up":{"uv":[24,42,12,30],"texture":0},"down":{"uv":[36,30,24,42],"texture":0}},"type":"cube","uuid":"2f7bd122-e685-5397-518a-d566774b5f7f"},{"name":"c","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-5,25,-5],"to":[5,48,5],"autouv":0,"color":7,"origin":[0,26,0],"uv_offset":[48,20],"faces":{"north":{"uv":[58,30,68,53],"texture":0},"east":{"uv":[48,30,58,53],"texture":0},"south":{"uv":[78,30,88,53],"texture":0},"west":{"uv":[68,30,78,53],"texture":0},"up":{"uv":[68,30,58,20],"texture":0},"down":{"uv":[78,20,68,30],"texture":0}},"type":"cube","uuid":"a32e4c53-99d7-93d5-23f7-ae163af4dae5"},{"name":"d","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-4,45,-4],"to":[4,66,4],"autouv":0,"color":7,"origin":[0,26,0],"uv_offset":[40,53],"faces":{"north":{"uv":[48,61,56,82],"texture":0},"east":{"uv":[40,61,48,82],"texture":0},"south":{"uv":[64,61,72,82],"texture":0},"west":{"uv":[56,61,64,82],"texture":0},"up":{"uv":[56,61,48,53],"texture":0},"down":{"uv":[64,53,56,61],"texture":0}},"type":"cube","uuid":"a4c5b57f-c555-154e-e673-7e114f6e5a21"},{"name":"e","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-3,64,-3],"to":[3,86,3],"autouv":0,"color":7,"origin":[-1.1793215979571126e-16,26,-4.382008627821946e-33],"uv_offset":[0,61],"faces":{"north":{"uv":[6,67,12,89],"texture":0},"east":{"uv":[0,67,6,89],"texture":0},"south":{"uv":[18,67,24,89],"texture":0},"west":{"uv":[12,67,18,89],"texture":0},"up":{"uv":[12,67,6,61],"texture":0},"down":{"uv":[18,61,12,67],"texture":0}},"type":"cube","uuid":"c8af392a-28ea-8a9f-cc0f-e8f19558f993"},{"name":"f","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-3,85,-3],"to":[3,100,3],"autouv":0,"color":7,"origin":[-2.620714662126916e-16,47,-9.737796950715435e-33],"uv_offset":[72,53],"faces":{"north":{"uv":[78,59,84,74],"texture":0},"east":{"uv":[72,59,78,74],"texture":0},"south":{"uv":[90,59,96,74],"texture":0},"west":{"uv":[84,59,90,74],"texture":0},"up":{"uv":[84,59,78,53],"texture":0},"down":{"uv":[90,53,84,59],"texture":0}},"type":"cube","uuid":"1d4836bc-6b01-9b8b-2958-1280f80f91b7"},{"name":"g","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-2.5999999999999996,99,-2.5999999999999996],"to":[2.5999999999999996,114,2.599999999999998],"autouv":0,"color":7,"origin":[-3.6034826604245074e-16,0,1],"uv_offset":[56,0],"faces":{"north":{"uv":[61.199999999999996,5.1999999999999975,66.39999999999999,20.199999999999996],"texture":0},"east":{"uv":[56,5.1999999999999975,61.199999999999996,20.199999999999996],"texture":0},"south":{"uv":[71.6,5.1999999999999975,76.8,20.199999999999996],"texture":0},"west":{"uv":[66.39999999999999,5.1999999999999975,71.6,20.199999999999996],"texture":0},"up":{"uv":[66.39999999999999,5.1999999999999975,61.199999999999996,0],"texture":0},"down":{"uv":[71.6,0,66.39999999999999,5.1999999999999975],"texture":0}},"type":"cube","uuid":"4001f6b1-caf3-d76f-2d5b-8636c85cd0fb"},{"name":"h","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-2.0999999999999996,113,-2.0999999999999996],"to":[2.0999999999999996,128,2.099999999999998],"autouv":0,"color":7,"origin":[-4.455214925615753e-16,14,1],"uv_offset":[24,61],"faces":{"north":{"uv":[28.199999999999996,65.2,32.4,80.19999999999999],"texture":0},"east":{"uv":[24,65.2,28.199999999999996,80.19999999999999],"texture":0},"south":{"uv":[36.599999999999994,65.2,40.8,80.19999999999999],"texture":0},"west":{"uv":[32.4,65.2,36.599999999999994,80.19999999999999],"texture":0},"up":{"uv":[32.4,65.2,28.199999999999996,61],"texture":0},"down":{"uv":[36.599999999999994,61,32.4,65.2],"texture":0}},"type":"cube","uuid":"4ba83545-3fcf-726d-033c-a1ac411a4ae3"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[3,-0.6703481918674496,-7.772537561589672],"to":[17,-0.6703481918674496,6.227462438410338],"autouv":0,"color":0,"rotation":[-89.99999999999994,-6.3611093629270335e-15,45],"origin":[3,-0.6703481918674496,-7.772537561589672],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.00000000000001,114.00000000000001,14.00000000000001],"texture":0},"east":{"uv":[86,14.00000000000001,100.00000000000001,14.00000000000001],"texture":0},"south":{"uv":[128.00000000000003,14.00000000000001,142.00000000000003,14.00000000000001],"texture":0},"west":{"uv":[114.00000000000001,14.00000000000001,128.00000000000003,14.00000000000001],"texture":0},"up":{"uv":[114.00000000000001,14.00000000000001,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.00000000000001],"texture":0}},"type":"cube","uuid":"43754e70-3fcd-b60a-62ee-01eadc22607a"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[3,-0.6703481918674496,-7.772537561589672],"to":[17,-0.6703481918674496,6.227462438410338],"autouv":0,"color":0,"rotation":[-180,-45.000000000000156,90],"origin":[3,-0.6703481918674496,-7.772537561589672],"uv_offset":[86,0],"faces":{"north":{"uv":[100,14.000000000000002,114,14.000000000000002],"texture":0},"east":{"uv":[86,14.000000000000002,100,14.000000000000002],"texture":0},"south":{"uv":[128,14.000000000000002,142,14.000000000000002],"texture":0},"west":{"uv":[114,14.000000000000002,128,14.000000000000002],"texture":0},"up":{"uv":[114,14.000000000000002,100,0],"texture":0},"down":{"uv":[128,0,114,14.000000000000002],"texture":0}},"type":"cube","uuid":"e5054bff-a980-a8dc-0221-58208085978f"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[5,-0.6703481918674511,4.227462438410328],"to":[19,-0.6703481918674511,18.22746243841034],"autouv":0,"color":0,"rotation":[-180,-45.000000000000156,90],"origin":[5,-0.6703481918674511,4.227462438410328],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.00000000000001,114.00000000000001,14.00000000000001],"texture":0},"east":{"uv":[86,14.00000000000001,100.00000000000001,14.00000000000001],"texture":0},"south":{"uv":[128.00000000000003,14.00000000000001,142.00000000000003,14.00000000000001],"texture":0},"west":{"uv":[114.00000000000001,14.00000000000001,128.00000000000003,14.00000000000001],"texture":0},"up":{"uv":[114.00000000000001,14.00000000000001,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.00000000000001],"texture":0}},"type":"cube","uuid":"13ebb478-c830-3a6e-858d-2269a5ea0642"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[5,-0.6703481918674511,4.227462438410328],"to":[19,-0.6703481918674511,18.22746243841034],"autouv":0,"color":0,"rotation":[-89.99999999999994,-6.3611093629270335e-15,45],"origin":[5,-0.6703481918674511,4.227462438410328],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.00000000000001,114.00000000000001,14.00000000000001],"texture":0},"east":{"uv":[86,14.00000000000001,100.00000000000001,14.00000000000001],"texture":0},"south":{"uv":[128.00000000000003,14.00000000000001,142.00000000000003,14.00000000000001],"texture":0},"west":{"uv":[114.00000000000001,14.00000000000001,128.00000000000003,14.00000000000001],"texture":0},"up":{"uv":[114.00000000000001,14.00000000000001,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.00000000000001],"texture":0}},"type":"cube","uuid":"2a87966b-98f6-fe32-0698-06d07dd0caa9"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-6,-0.6703481918674511,4.227462438410328],"to":[8,-0.6703481918674511,18.22746243841034],"autouv":0,"color":0,"rotation":[-180,-45.000000000000156,90],"origin":[-6,-0.6703481918674511,4.227462438410328],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.00000000000001,114.00000000000001,14.00000000000001],"texture":0},"east":{"uv":[86,14.00000000000001,100.00000000000001,14.00000000000001],"texture":0},"south":{"uv":[128.00000000000003,14.00000000000001,142.00000000000003,14.00000000000001],"texture":0},"west":{"uv":[114.00000000000001,14.00000000000001,128.00000000000003,14.00000000000001],"texture":0},"up":{"uv":[114.00000000000001,14.00000000000001,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.00000000000001],"texture":0}},"type":"cube","uuid":"b2e2dbad-e1e3-ecb6-41d7-21cb1908deb8"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-6,-0.6703481918674511,4.227462438410328],"to":[8,-0.6703481918674511,18.22746243841034],"autouv":0,"color":0,"rotation":[-89.99999999999994,-6.3611093629270335e-15,45],"origin":[-6,-0.6703481918674511,4.227462438410328],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.00000000000001,114.00000000000001,14.00000000000001],"texture":0},"east":{"uv":[86,14.00000000000001,100.00000000000001,14.00000000000001],"texture":0},"south":{"uv":[128.00000000000003,14.00000000000001,142.00000000000003,14.00000000000001],"texture":0},"west":{"uv":[114.00000000000001,14.00000000000001,128.00000000000003,14.00000000000001],"texture":0},"up":{"uv":[114.00000000000001,14.00000000000001,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.00000000000001],"texture":0}},"type":"cube","uuid":"f93aa31f-c197-3e66-58ad-a1c638ecce97"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-6,-0.6703481918674499,-6.772537561589672],"to":[8,-0.6703481918674499,7.227462438410338],"autouv":0,"color":0,"rotation":[-180,-45.000000000000156,90],"origin":[-6,-0.6703481918674499,-6.772537561589672],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.00000000000001,114.00000000000001,14.00000000000001],"texture":0},"east":{"uv":[86,14.00000000000001,100.00000000000001,14.00000000000001],"texture":0},"south":{"uv":[128.00000000000003,14.00000000000001,142.00000000000003,14.00000000000001],"texture":0},"west":{"uv":[114.00000000000001,14.00000000000001,128.00000000000003,14.00000000000001],"texture":0},"up":{"uv":[114.00000000000001,14.00000000000001,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.00000000000001],"texture":0}},"type":"cube","uuid":"c1ca47a6-f9ba-555a-cc32-58d55b8809b9"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-6,-0.6703481918674499,-6.772537561589672],"to":[8,-0.6703481918674499,7.227462438410338],"autouv":0,"color":0,"rotation":[-89.99999999999994,-6.3611093629270335e-15,45],"origin":[-6,-0.6703481918674499,-6.772537561589672],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.00000000000001,114.00000000000001,14.00000000000001],"texture":0},"east":{"uv":[86,14.00000000000001,100.00000000000001,14.00000000000001],"texture":0},"south":{"uv":[128.00000000000003,14.00000000000001,142.00000000000003,14.00000000000001],"texture":0},"west":{"uv":[114.00000000000001,14.00000000000001,128.00000000000003,14.00000000000001],"texture":0},"up":{"uv":[114.00000000000001,14.00000000000001,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.00000000000001],"texture":0}},"type":"cube","uuid":"b8b98b2d-632b-3acd-29a8-55efa4850547"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[3,-0.6703481918674496,-7.772537561589672],"to":[17,-0.6703481918674496,6.227462438410338],"autouv":0,"color":0,"rotation":[-180,-45.000000000000156,90],"origin":[3,-0.6703481918674496,-7.772537561589672],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.00000000000001,114.00000000000001,14.00000000000001],"texture":0},"east":{"uv":[86,14.00000000000001,100.00000000000001,14.00000000000001],"texture":0},"south":{"uv":[128.00000000000003,14.00000000000001,142.00000000000003,14.00000000000001],"texture":0},"west":{"uv":[114.00000000000001,14.00000000000001,128.00000000000003,14.00000000000001],"texture":0},"up":{"uv":[114.00000000000001,14.00000000000001,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.00000000000001],"texture":0}},"type":"cube","uuid":"e3aaee1e-958c-370c-a10d-94e142df629c"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[3,-0.6703481918674496,-7.772537561589672],"to":[17,-0.6703481918674496,6.227462438410338],"autouv":0,"color":0,"rotation":[-89.99999999999994,-6.3611093629270335e-15,45],"origin":[3,-0.6703481918674496,-7.772537561589672],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.00000000000001,114.00000000000001,14.00000000000001],"texture":0},"east":{"uv":[86,14.00000000000001,100.00000000000001,14.00000000000001],"texture":0},"south":{"uv":[128.00000000000003,14.00000000000001,142.00000000000003,14.00000000000001],"texture":0},"west":{"uv":[114.00000000000001,14.00000000000001,128.00000000000003,14.00000000000001],"texture":0},"up":{"uv":[114.00000000000001,14.00000000000001,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.00000000000001],"texture":0}},"type":"cube","uuid":"6bf59ab2-03d6-2995-17bc-76e27e52bf6c"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[5,-0.6703481918674511,4.227462438410328],"to":[19,-0.6703481918674511,18.22746243841034],"autouv":0,"color":0,"rotation":[-180,-45.000000000000156,90],"origin":[5,-0.6703481918674511,4.227462438410328],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.00000000000001,114.00000000000001,14.00000000000001],"texture":0},"east":{"uv":[86,14.00000000000001,100.00000000000001,14.00000000000001],"texture":0},"south":{"uv":[128.00000000000003,14.00000000000001,142.00000000000003,14.00000000000001],"texture":0},"west":{"uv":[114.00000000000001,14.00000000000001,128.00000000000003,14.00000000000001],"texture":0},"up":{"uv":[114.00000000000001,14.00000000000001,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.00000000000001],"texture":0}},"type":"cube","uuid":"666a373c-9dd2-0cfc-c1b7-13a5fe7fe392"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[5,-0.6703481918674511,4.227462438410328],"to":[19,-0.6703481918674511,18.22746243841034],"autouv":0,"color":0,"rotation":[-89.99999999999994,-6.3611093629270335e-15,45],"origin":[5,-0.6703481918674511,4.227462438410328],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.00000000000001,114.00000000000001,14.00000000000001],"texture":0},"east":{"uv":[86,14.00000000000001,100.00000000000001,14.00000000000001],"texture":0},"south":{"uv":[128.00000000000003,14.00000000000001,142.00000000000003,14.00000000000001],"texture":0},"west":{"uv":[114.00000000000001,14.00000000000001,128.00000000000003,14.00000000000001],"texture":0},"up":{"uv":[114.00000000000001,14.00000000000001,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.00000000000001],"texture":0}},"type":"cube","uuid":"ebaca892-208f-1ec8-a7b0-4b8bf692d869"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-6,-0.6703481918674511,4.227462438410328],"to":[8,-0.6703481918674511,18.22746243841034],"autouv":0,"color":0,"rotation":[-180,-45.000000000000156,90],"origin":[-6,-0.6703481918674511,4.227462438410328],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.00000000000001,114.00000000000001,14.00000000000001],"texture":0},"east":{"uv":[86,14.00000000000001,100.00000000000001,14.00000000000001],"texture":0},"south":{"uv":[128.00000000000003,14.00000000000001,142.00000000000003,14.00000000000001],"texture":0},"west":{"uv":[114.00000000000001,14.00000000000001,128.00000000000003,14.00000000000001],"texture":0},"up":{"uv":[114.00000000000001,14.00000000000001,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.00000000000001],"texture":0}},"type":"cube","uuid":"02bb3148-7ad4-1fbe-f233-fcd02c1f00f4"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-6,-0.6703481918674511,4.227462438410328],"to":[8,-0.6703481918674511,18.22746243841034],"autouv":0,"color":0,"rotation":[-89.99999999999994,-6.3611093629270335e-15,45],"origin":[-6,-0.6703481918674511,4.227462438410328],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.00000000000001,114.00000000000001,14.00000000000001],"texture":0},"east":{"uv":[86,14.00000000000001,100.00000000000001,14.00000000000001],"texture":0},"south":{"uv":[128.00000000000003,14.00000000000001,142.00000000000003,14.00000000000001],"texture":0},"west":{"uv":[114.00000000000001,14.00000000000001,128.00000000000003,14.00000000000001],"texture":0},"up":{"uv":[114.00000000000001,14.00000000000001,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.00000000000001],"texture":0}},"type":"cube","uuid":"ea8790f9-39c1-9607-7faf-4054f0782315"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-6,-0.6703481918674499,-6.772537561589672],"to":[8,-0.6703481918674499,7.227462438410338],"autouv":0,"color":0,"rotation":[-180,-45.000000000000156,90],"origin":[-6,-0.6703481918674499,-6.772537561589672],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.00000000000001,114.00000000000001,14.00000000000001],"texture":0},"east":{"uv":[86,14.00000000000001,100.00000000000001,14.00000000000001],"texture":0},"south":{"uv":[128.00000000000003,14.00000000000001,142.00000000000003,14.00000000000001],"texture":0},"west":{"uv":[114.00000000000001,14.00000000000001,128.00000000000003,14.00000000000001],"texture":0},"up":{"uv":[114.00000000000001,14.00000000000001,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.00000000000001],"texture":0}},"type":"cube","uuid":"42f35de8-e9da-314a-ea8b-4ec3162ccc7b"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-6,-0.6703481918674499,-6.772537561589672],"to":[8,-0.6703481918674499,7.227462438410338],"autouv":0,"color":0,"rotation":[-89.99999999999994,-6.3611093629270335e-15,45],"origin":[-6,-0.6703481918674499,-6.772537561589672],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.00000000000001,114.00000000000001,14.00000000000001],"texture":0},"east":{"uv":[86,14.00000000000001,100.00000000000001,14.00000000000001],"texture":0},"south":{"uv":[128.00000000000003,14.00000000000001,142.00000000000003,14.00000000000001],"texture":0},"west":{"uv":[114.00000000000001,14.00000000000001,128.00000000000003,14.00000000000001],"texture":0},"up":{"uv":[114.00000000000001,14.00000000000001,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.00000000000001],"texture":0}},"type":"cube","uuid":"2a148807-d884-e05b-680e-36ba4ce450e5"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-4,40.62422566484149,-3.3435187525409757],"to":[10,40.62422566484149,10.656481247459041],"autouv":0,"color":0,"rotation":[-180,-45.000000000000156,90],"origin":[-4,40.62422566484149,-3.3435187525409757],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.00000000000001,114.00000000000001,14.00000000000001],"texture":0},"east":{"uv":[86,14.00000000000001,100.00000000000001,14.00000000000001],"texture":0},"south":{"uv":[128.00000000000003,14.00000000000001,142.00000000000003,14.00000000000001],"texture":0},"west":{"uv":[114.00000000000001,14.00000000000001,128.00000000000003,14.00000000000001],"texture":0},"up":{"uv":[114.00000000000001,14.00000000000001,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.00000000000001],"texture":0}},"type":"cube","uuid":"e0119d00-cad6-3842-bbe5-6885628ff35e"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-4,40.62422566484149,-3.3435187525409757],"to":[10,40.62422566484149,10.656481247459041],"autouv":0,"color":0,"rotation":[-89.99999999999994,-6.3611093629270335e-15,45],"origin":[-4,40.62422566484149,-3.3435187525409757],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.00000000000001,114.00000000000001,14.00000000000001],"texture":0},"east":{"uv":[86,14.00000000000001,100.00000000000001,14.00000000000001],"texture":0},"south":{"uv":[128.00000000000003,14.00000000000001,142.00000000000003,14.00000000000001],"texture":0},"west":{"uv":[114.00000000000001,14.00000000000001,128.00000000000003,14.00000000000001],"texture":0},"up":{"uv":[114.00000000000001,14.00000000000001,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.00000000000001],"texture":0}},"type":"cube","uuid":"abe6d75d-fccf-7b97-e8ca-156f68f30b76"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[2,60.13549813476108,3.0055163398616545],"to":[16,60.13549813476108,17.005516339861675],"autouv":0,"color":0,"rotation":[-180,-45.000000000000156,90],"origin":[2,60.13549813476108,3.0055163398616545],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.000000000000018,114.00000000000001,14.000000000000018],"texture":0},"east":{"uv":[86,14.000000000000018,100.00000000000001,14.000000000000018],"texture":0},"south":{"uv":[128.00000000000003,14.000000000000018,142.00000000000003,14.000000000000018],"texture":0},"west":{"uv":[114.00000000000001,14.000000000000018,128.00000000000003,14.000000000000018],"texture":0},"up":{"uv":[114.00000000000001,14.000000000000018,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.000000000000018],"texture":0}},"type":"cube","uuid":"bf2e26ba-937a-73c8-3ab2-9240c2690b0b"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[2,60.13549813476108,3.0055163398616545],"to":[16,60.13549813476108,17.005516339861675],"autouv":0,"color":0,"rotation":[-89.99999999999994,-6.3611093629270335e-15,45],"origin":[2,60.13549813476108,3.0055163398616545],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000001,14.000000000000018,114.00000000000001,14.000000000000018],"texture":0},"east":{"uv":[86,14.000000000000018,100.00000000000001,14.000000000000018],"texture":0},"south":{"uv":[128.00000000000003,14.000000000000018,142.00000000000003,14.000000000000018],"texture":0},"west":{"uv":[114.00000000000001,14.000000000000018,128.00000000000003,14.000000000000018],"texture":0},"up":{"uv":[114.00000000000001,14.000000000000018,100.00000000000001,0],"texture":0},"down":{"uv":[128,0,114.00000000000001,14.000000000000018],"texture":0}},"type":"cube","uuid":"61246d8c-3a21-f2e2-2b5e-656b618b9315"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[2.0000000000000004,89.02514437942168,0.47799980017960014],"to":[16,89.02514437942168,14.47799980017963],"autouv":0,"color":0,"rotation":[-180,-45.000000000000156,90],"origin":[2.0000000000000004,89.02514437942168,0.47799980017960014],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000003,14.000000000000021,114.00000000000003,14.000000000000021],"texture":0},"east":{"uv":[86,14.000000000000021,100.00000000000003,14.000000000000021],"texture":0},"south":{"uv":[128.00000000000006,14.000000000000021,142.00000000000006,14.000000000000021],"texture":0},"west":{"uv":[114.00000000000003,14.000000000000021,128.00000000000006,14.000000000000021],"texture":0},"up":{"uv":[114.00000000000003,14.000000000000021,100.00000000000003,0],"texture":0},"down":{"uv":[128.00000000000003,0,114.00000000000003,14.000000000000021],"texture":0}},"type":"cube","uuid":"ff5dbdd3-d6eb-27ba-e1ab-c1a0f550fd3e"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[2.0000000000000004,89.02514437942168,0.47799980017960014],"to":[16,89.02514437942168,14.47799980017963],"autouv":0,"color":0,"rotation":[-89.99999999999994,-6.3611093629270335e-15,45],"origin":[2.0000000000000004,89.02514437942168,0.47799980017960014],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000003,14.000000000000021,114.00000000000003,14.000000000000021],"texture":0},"east":{"uv":[86,14.000000000000021,100.00000000000003,14.000000000000021],"texture":0},"south":{"uv":[128.00000000000006,14.000000000000021,142.00000000000006,14.000000000000021],"texture":0},"west":{"uv":[114.00000000000003,14.000000000000021,128.00000000000006,14.000000000000021],"texture":0},"up":{"uv":[114.00000000000003,14.000000000000021,100.00000000000003,0],"texture":0},"down":{"uv":[128.00000000000003,0,114.00000000000003,14.000000000000021],"texture":0}},"type":"cube","uuid":"8e0f248f-1228-4a38-2a74-cb0d23718fdc"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-22.881490097225765,145.88837332325411,6.8760974475566785],"to":[-8.881490097225756,145.88837332325411,20.876097447556763],"autouv":0,"color":0,"rotation":[-180,-45.000000000000156,90],"origin":[-36.04320341964982,110.62188448501286,-2.507082580585397],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000009,14.000000000000085,114.0000000000001,14.000000000000085],"texture":0},"east":{"uv":[86,14.000000000000085,100.00000000000009,14.000000000000085],"texture":0},"south":{"uv":[128.00000000000017,14.000000000000085,142.0000000000002,14.000000000000085],"texture":0},"west":{"uv":[114.0000000000001,14.000000000000085,128.00000000000017,14.000000000000085],"texture":0},"up":{"uv":[114.0000000000001,14.000000000000085,100.00000000000009,0],"texture":0},"down":{"uv":[128.0000000000001,0,114.0000000000001,14.000000000000085],"texture":0}},"type":"cube","uuid":"817666d5-286f-6161-8446-3970704a3265"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[0.16641666179338577,107.95005796968688,-16.171809311462464],"to":[14.166416661793397,107.95005796968688,-2.17180931146247],"autouv":0,"color":0,"rotation":[-89.99999999999994,-6.3611093629270335e-15,45],"origin":[-36.04320341964982,110.62188448501286,-2.507082580585397],"uv_offset":[86,0],"faces":{"north":{"uv":[100,13.999999999999993,114,13.999999999999993],"texture":0},"east":{"uv":[86,13.999999999999993,100,13.999999999999993],"texture":0},"south":{"uv":[128,13.999999999999993,142,13.999999999999993],"texture":0},"west":{"uv":[114,13.999999999999993,128,13.999999999999993],"texture":0},"up":{"uv":[114,13.999999999999993,100,0],"texture":0},"down":{"uv":[128,0,114,13.999999999999993],"texture":0}},"type":"cube","uuid":"5f3e16dd-eed6-733e-e1db-f1934b558e88"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-22.881490097225765,145.88837332325411,6.8760974475566785],"to":[-8.881490097225756,145.88837332325411,20.876097447556763],"autouv":0,"color":0,"rotation":[-180,-45.000000000000156,90],"origin":[-36.04320341964982,110.62188448501286,-2.507082580585397],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000009,14.000000000000085,114.00000000000009,14.000000000000085],"texture":0},"east":{"uv":[86,14.000000000000085,100.00000000000009,14.000000000000085],"texture":0},"south":{"uv":[128.00000000000017,14.000000000000085,142.00000000000017,14.000000000000085],"texture":0},"west":{"uv":[114.00000000000009,14.000000000000085,128.00000000000017,14.000000000000085],"texture":0},"up":{"uv":[114.00000000000009,14.000000000000085,100.00000000000009,0],"texture":0},"down":{"uv":[128.0000000000001,0,114.00000000000009,14.000000000000085],"texture":0}},"type":"cube","uuid":"573cf719-c66a-d73e-148d-d14e9617d0ac"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[0.16641666179338577,107.95005796968688,-16.171809311462464],"to":[14.166416661793397,107.95005796968688,-2.17180931146247],"autouv":0,"color":0,"rotation":[-89.99999999999994,-6.3611093629270335e-15,45],"origin":[-36.04320341964982,110.62188448501286,-2.507082580585397],"uv_offset":[86,0],"faces":{"north":{"uv":[100,13.999999999999995,114,13.999999999999995],"texture":0},"east":{"uv":[86,13.999999999999995,100,13.999999999999995],"texture":0},"south":{"uv":[128,13.999999999999995,142,13.999999999999995],"texture":0},"west":{"uv":[114,13.999999999999995,128,13.999999999999995],"texture":0},"up":{"uv":[114,13.999999999999995,100,0],"texture":0},"down":{"uv":[128,0,114,13.999999999999995],"texture":0}},"type":"cube","uuid":"e4e3e2ae-0684-ee78-4ec9-33f0f9461b19"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-22.881490097225765,145.88837332325411,6.8760974475566785],"to":[-8.881490097225756,145.88837332325411,20.876097447556763],"autouv":0,"color":0,"rotation":[-180,-45.000000000000156,90],"origin":[-36.04320341964982,110.62188448501286,-2.507082580585397],"uv_offset":[86,0],"faces":{"north":{"uv":[100.00000000000009,14.000000000000085,114.00000000000009,14.000000000000085],"texture":0},"east":{"uv":[86,14.000000000000085,100.00000000000009,14.000000000000085],"texture":0},"south":{"uv":[128.00000000000017,14.000000000000085,142.00000000000017,14.000000000000085],"texture":0},"west":{"uv":[114.00000000000009,14.000000000000085,128.00000000000017,14.000000000000085],"texture":0},"up":{"uv":[114.00000000000009,14.000000000000085,100.00000000000009,0],"texture":0},"down":{"uv":[128.0000000000001,0,114.00000000000009,14.000000000000085],"texture":0}},"type":"cube","uuid":"c59ea2f3-dbd7-5b92-4640-4acc2121a8b2"},{"name":"cube","box_uv":true,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[0.16641666179338577,107.95005796968688,-16.171809311462464],"to":[14.166416661793397,107.95005796968688,-2.17180931146247],"autouv":0,"color":0,"rotation":[-89.99999999999994,-6.3611093629270335e-15,45],"origin":[-36.04320341964982,110.62188448501286,-2.507082580585397],"uv_offset":[86,0],"faces":{"north":{"uv":[100,13.999999999999995,114,13.999999999999995],"texture":0},"east":{"uv":[86,13.999999999999995,100,13.999999999999995],"texture":0},"south":{"uv":[128,13.999999999999995,142,13.999999999999995],"texture":0},"west":{"uv":[114,13.999999999999995,128,13.999999999999995],"texture":0},"up":{"uv":[114,13.999999999999995,100,0],"texture":0},"down":{"uv":[128,0,114,13.999999999999995],"texture":0}},"type":"cube","uuid":"3bb469e2-e5a5-7cfc-3e9c-6fd43b8a3ca0"}],"outliner":[{"name":"bone_a","origin":[0,0,0],"rotation":[-7.951386703658792e-15,0,0],"color":0,"uuid":"836e9354-0309-70b5-cb60-e7f38118e743","export":true,"mirror_uv":false,"isOpen":true,"locked":false,"visibility":true,"autouv":0,"children":["43deca74-ef0e-9496-f060-a569a5f66a3c",{"name":"flower_4","origin":[-6,-0.6703481918674499,-6.772537561589672],"rotation":[-91.26598229300457,52.08446097361874,-64.05377573888674],"color":0,"uuid":"2ea8b2cf-a2c0-c454-2fb5-d9bc522a921c","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["c1ca47a6-f9ba-555a-cc32-58d55b8809b9","b8b98b2d-632b-3acd-29a8-55efa4850547",{"name":"flower_8","origin":[-6,-0.6703481918674499,-6.772537561589672],"rotation":[-167.64364206893785,46.98788282368353,-125.90165504433261],"color":0,"uuid":"02ef3343-dfcc-9af4-29a2-6dc427c70898","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["42f35de8-e9da-314a-ea8b-4ec3162ccc7b","2a148807-d884-e05b-680e-36ba4ce450e5"]}]},{"name":"flower_3","origin":[-6,-0.6703481918674511,4.227462438410328],"rotation":[137.96042965468234,-13.430580338215425,-145.38236406964168],"color":0,"uuid":"2fc2f20b-05ce-0410-dd40-8d55ec1d814a","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["b2e2dbad-e1e3-ecb6-41d7-21cb1908deb8","f93aa31f-c197-3e66-58ad-a1c638ecce97",{"name":"flower_7","origin":[-6,-0.6703481918674511,4.227462438410328],"rotation":[137.96042965468234,-13.430580338215425,-145.38236406964168],"color":0,"uuid":"3c08633a-6d30-3951-f49a-b815826de0cd","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["02bb3148-7ad4-1fbe-f233-fcd02c1f00f4","ea8790f9-39c1-9607-7faf-4054f0782315"]}]},{"name":"flower_2","origin":[5,-0.6703481918674511,4.227462438410328],"rotation":[72.75299322235041,-55.44873947197246,-103.02095072899253],"color":0,"uuid":"8cc35e91-dbf2-1e20-efc6-d0c6dc9e1e32","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["13ebb478-c830-3a6e-858d-2269a5ea0642","2a87966b-98f6-fe32-0698-06d07dd0caa9",{"name":"flower_6","origin":[5,-0.6703481918674511,4.227462438410328],"rotation":[72.75299322235041,-55.44873947197246,-103.02095072899253],"color":0,"uuid":"93297f86-e3ca-4d71-a6c8-298ff77bbb1a","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["666a373c-9dd2-0cfc-c1b7-13a5fe7fe392","ebaca892-208f-1ec8-a7b0-4b8bf692d869"]}]},{"name":"flower_1","origin":[3,-0.6703481918674496,-7.772537561589672],"rotation":[-34.96649542120608,3.067430398401471,-33.59739024796235],"color":0,"uuid":"59363886-4b5a-2a3c-75cd-8d6276de69b3","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["e5054bff-a980-a8dc-0221-58208085978f","43754e70-3fcd-b60a-62ee-01eadc22607a",{"name":"flower_5","origin":[3,-0.6703481918674496,-7.772537561589672],"rotation":[-34.96649542120608,3.067430398401471,-33.59739024796235],"color":0,"uuid":"0265d420-aeaa-b538-1252-89df12b0f9b6","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["e3aaee1e-958c-370c-a10d-94e142df629c","6bf59ab2-03d6-2995-17bc-76e27e52bf6c"]}]},{"name":"bone_b","origin":[0,9,0],"rotation":[15,0,0],"color":0,"uuid":"aa6d7f64-89b9-a900-a3dc-3a9fcdc22618","export":true,"mirror_uv":false,"isOpen":true,"locked":false,"visibility":true,"autouv":0,"children":["2f7bd122-e685-5397-518a-d566774b5f7f",{"name":"bone_c","origin":[0,25,0],"rotation":[-6.599650964036798e-14,0,0],"color":0,"uuid":"7b4502a2-fe6a-53ad-2f9f-391f31027828","export":true,"mirror_uv":false,"isOpen":true,"locked":false,"visibility":true,"autouv":0,"children":[{"name":"flower_9","origin":[-4,40.62422566484149,-3.3435187525409757],"rotation":[-167.64364206893785,46.987882823683556,-140.9016550443326],"color":0,"uuid":"5ba196d8-95ae-bace-28cb-48bdb2d6cbac","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["e0119d00-cad6-3842-bbe5-6885628ff35e","abe6d75d-fccf-7b97-e8ca-156f68f30b76"]},"a32e4c53-99d7-93d5-23f7-ae163af4dae5",{"name":"bone_d","origin":[0,43,0],"rotation":[10.000000000000005,2.1289409148363706e-15,3.753897236198147e-16],"color":0,"uuid":"286e227f-8a7a-cfd3-27a0-86aed0c10bb7","export":true,"mirror_uv":false,"isOpen":true,"locked":false,"visibility":true,"autouv":0,"children":[{"name":"flower_10","origin":[2,60.13549813476108,3.0055163398616545],"rotation":[163.90070263038493,34.045350837165316,140.2721846400254],"color":0,"uuid":"1ed2c36a-c359-91ea-40ba-572a6c8757f7","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["bf2e26ba-937a-73c8-3ab2-9240c2690b0b","61246d8c-3a21-f2e2-2b5e-656b618b9315"]},"a4c5b57f-c555-154e-e673-7e114f6e5a21",{"name":"bone_e","origin":[0,61,0],"rotation":[-10,2.1289409148363706e-15,3.753897236198147e-16],"color":0,"uuid":"960ce6d7-f146-7814-e0c9-b11b5b43956a","export":true,"mirror_uv":false,"isOpen":true,"locked":false,"visibility":true,"autouv":0,"children":["c8af392a-28ea-8a9f-cc0f-e8f19558f993",{"name":"bone_f","origin":[0,83,0],"rotation":[-10,2.1289409148363706e-15,3.753897236198147e-16],"color":0,"uuid":"3f876c03-f9b2-1dac-08bc-990abce00d76","export":true,"mirror_uv":false,"isOpen":true,"locked":false,"visibility":true,"autouv":0,"children":["1d4836bc-6b01-9b8b-2958-1280f80f91b7",{"name":"flower_11","origin":[2.0000000000000004,89.02514437942168,0.47799980017960014],"rotation":[-158.0615625451014,-19.935030254861065,145.712572187446],"color":0,"uuid":"42c04c52-8414-cc27-f1a2-bce1e6fdc48c","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["ff5dbdd3-d6eb-27ba-e1ab-c1a0f550fd3e","8e0f248f-1228-4a38-2a74-cb0d23718fdc"]},{"name":"bone_g","origin":[0,98,0],"rotation":[-17.5,2.1289409148363706e-15,3.753897236198147e-16],"color":0,"uuid":"87fdafb8-862b-0fc4-f194-483e4c6ab1d2","export":true,"mirror_uv":false,"isOpen":true,"locked":false,"visibility":true,"autouv":0,"children":["4001f6b1-caf3-d76f-2d5b-8636c85cd0fb",{"name":"bone_h","origin":[0,111,0],"rotation":[-15,2.1289409148363706e-15,3.753897236198147e-16],"color":0,"uuid":"bd8fb7b0-ef45-7564-a12a-05cb47421493","export":true,"mirror_uv":false,"isOpen":true,"locked":false,"visibility":true,"autouv":0,"children":["4ba83545-3fcf-726d-033c-a1ac411a4ae3",{"name":"flower_12","origin":[-4.852359674770397e-16,126.1462282992986,0.6365198729059891],"rotation":[-158.06156254510137,-19.93503025486101,170.7125721874457],"color":0,"uuid":"faa91113-8626-24ec-baa2-9003ea1e0eda","export":true,"mirror_uv":false,"isOpen":true,"locked":false,"visibility":true,"autouv":0,"children":["817666d5-286f-6161-8446-3970704a3265","5f3e16dd-eed6-733e-e1db-f1934b558e88",{"name":"flower_13","origin":[-4.852359674770397e-16,126.1462282992986,0.6365198729059891],"rotation":[-158.06156254510134,-19.935030254861015,148.21257218744574],"color":0,"uuid":"749d951a-ecbc-4197-9c32-365bbb0a98a9","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["573cf719-c66a-d73e-148d-d14e9617d0ac","e4e3e2ae-0684-ee78-4ec9-33f0f9461b19"]},{"name":"flower_14","origin":[-4.852359674770397e-16,126.1462282992986,0.6365198729059891],"rotation":[111.22147558087971,59.83922506916349,99.72701164210366],"color":0,"uuid":"3c36c30a-bdb5-bede-7d1c-4073204f716c","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["c59ea2f3-dbd7-5b92-4640-4acc2121a8b2","3bb469e2-e5a5-7cfc-3e9c-6fd43b8a3ca0"]}]}]}]}]}]}]}]}]}]}],"textures":[{"path":"/home/sollace/Documents/GitRepos/minecraft_mods/Unicopia/assets/models/tentacle.png","name":"tentacle.png","folder":"block","namespace":"","id":"0","particle":false,"render_mode":"default","render_sides":"auto","frame_time":1,"frame_order_type":"loop","frame_order":"","frame_interpolate":false,"visible":true,"mode":"bitmap","saved":true,"uuid":"59729803-364e-77c5-ab9a-5c088a316d59","relative_path":"../tentacle.png","source":""}],"fabricOptions":{"header":"package com.example.mod;","entity":"Entity","render":"","members":""}}
\ No newline at end of file
diff --git a/assets/models/tentacle.png b/assets/models/tentacle.png
new file mode 100644
index 00000000..1026c1c0
Binary files /dev/null and b/assets/models/tentacle.png differ
diff --git a/assets/spectral_clock_0.xcf b/assets/spectral_clock_0.xcf
new file mode 100644
index 00000000..82bac275
Binary files /dev/null and b/assets/spectral_clock_0.xcf differ
diff --git a/src/main/java/com/minelittlepony/unicopia/Config.java b/src/main/java/com/minelittlepony/unicopia/Config.java
index c0f86374..3a128a0e 100644
--- a/src/main/java/com/minelittlepony/unicopia/Config.java
+++ b/src/main/java/com/minelittlepony/unicopia/Config.java
@@ -45,6 +45,16 @@ public class Config extends com.minelittlepony.common.util.settings.Config {
.addComment("Removes butterflies from spawning in your world")
.addComment("Turn this ON if you have another mod that adds butterflies.");
+ public final Setting simplifiedPortals = value("compatibility", "simplifiedPortals", false)
+ .addComment("Disables dynamic portal rendering");
+
+ public final Setting fancyPortalRefreshRate = value("client", "fancyPortalRefreshRate", -1L)
+ .addComment("Sets the refresh rate of portals when using fancy portal rendering")
+ .addComment("Set to -1 (default) for unlimited");
+
+ public final Setting maxPortalRecursion = value("client", "maxPortalRecursion", 2)
+ .addComment("Sets the maximum depth to reach when rendering portals through portals");
+
public Config() {
super(new HeirarchicalJsonConfigAdapter(new GsonBuilder()
.registerTypeAdapter(Race.class, RegistryTypeAdapter.of(Race.REGISTRY))
diff --git a/src/main/java/com/minelittlepony/unicopia/Debug.java b/src/main/java/com/minelittlepony/unicopia/Debug.java
index 2e5a9982..8cc8847e 100644
--- a/src/main/java/com/minelittlepony/unicopia/Debug.java
+++ b/src/main/java/com/minelittlepony/unicopia/Debug.java
@@ -1,33 +1,35 @@
package com.minelittlepony.unicopia;
+import java.util.Set;
+import java.util.concurrent.atomic.AtomicReference;
+import java.util.stream.Collectors;
+
+import com.google.common.collect.Sets;
import com.minelittlepony.unicopia.ability.magic.spell.trait.SpellTraits;
import com.minelittlepony.unicopia.entity.mob.AirBalloonEntity;
import com.minelittlepony.unicopia.entity.mob.UEntities;
import net.minecraft.entity.vehicle.BoatEntity;
import net.minecraft.registry.Registries;
+import net.minecraft.registry.tag.TagKey;
+import net.minecraft.util.Identifier;
import net.minecraft.world.World;
+import net.minecraft.world.dimension.DimensionTypes;
public interface Debug {
boolean SPELLBOOK_CHAPTERS = Boolean.getBoolean("unicopia.debug.spellbookChapters");
boolean CHECK_GAME_VALUES = Boolean.getBoolean("unicopia.debug.checkGameValues");
+ boolean CHECK_TRAIT_COVERAGE = Boolean.getBoolean("unicopia.debug.checkTraitCoverage");
- boolean[] TESTS_COMPLETE = {false};
+ AtomicReference LAST_TESTED_WORLD = new AtomicReference<>(null);
static void runTests(World world) {
- if (!CHECK_GAME_VALUES || TESTS_COMPLETE[0]) {
+ if (!CHECK_GAME_VALUES || !world.getDimensionKey().getValue().equals(DimensionTypes.OVERWORLD_ID) || (LAST_TESTED_WORLD.getAndSet(world) == world)) {
return;
}
- TESTS_COMPLETE[0] = true;
- try {
- Registries.ITEM.getEntrySet().forEach(entry -> {
- if (SpellTraits.of(entry.getValue()).isEmpty()) {
- // Unicopia.LOGGER.warn("No traits registered for item {}", entry.getKey());
- }
- });
- } catch (Throwable t) {
- throw new IllegalStateException("Tests failed", t);
+ if (CHECK_TRAIT_COVERAGE) {
+ testTraitCoverage();
}
try {
@@ -40,4 +42,30 @@ public interface Debug {
throw new IllegalStateException("Tests failed", t);
}
}
+
+ private static void testTraitCoverage() {
+ Registries.ITEM.getEntrySet().stream().collect(Collectors.toMap(
+ entry -> entry.getKey().getValue().getNamespace(),
+ Set::of,
+ Sets::union
+ )).forEach((namespace, entries) -> {
+ @SuppressWarnings("deprecation")
+ var unregistered = entries.stream()
+ .filter(entry -> !entry.getValue().getRegistryEntry().isIn(UTags.HAS_NO_TRAITS) && SpellTraits.of(entry.getValue()).isEmpty())
+ .map(entry -> {
+ String id = entry.getKey().getValue().toString();
+
+ return id + "(" + Registries.ITEM.streamTags()
+ .filter(entry.getValue().getRegistryEntry()::isIn)
+ .map(TagKey::id)
+ .map(Identifier::toString)
+ .collect(Collectors.joining(", ")) + ")";
+ })
+ .toList();
+
+ if (!unregistered.isEmpty()) {
+ Unicopia.LOGGER.warn("No traits registered for {} items in namepsace {} {}", unregistered.size(), namespace, String.join(",\r\n", unregistered));
+ }
+ });
+ }
}
diff --git a/src/main/java/com/minelittlepony/unicopia/EntityConvertable.java b/src/main/java/com/minelittlepony/unicopia/EntityConvertable.java
index d21ac152..6a0ffce7 100644
--- a/src/main/java/com/minelittlepony/unicopia/EntityConvertable.java
+++ b/src/main/java/com/minelittlepony/unicopia/EntityConvertable.java
@@ -25,7 +25,7 @@ public interface EntityConvertable extends WorldConvertable {
* Gets the center position where this caster is located.
*/
default Vec3d getOriginVector() {
- return asEntity().getPos();
+ return asEntity().getPos().add(0, asEntity().getHeight() * 0.5F, 0);
}
@Override
diff --git a/src/main/java/com/minelittlepony/unicopia/EquinePredicates.java b/src/main/java/com/minelittlepony/unicopia/EquinePredicates.java
index be559b38..9f055212 100644
--- a/src/main/java/com/minelittlepony/unicopia/EquinePredicates.java
+++ b/src/main/java/com/minelittlepony/unicopia/EquinePredicates.java
@@ -33,7 +33,7 @@ public interface EquinePredicates {
Predicate IS_CASTER = e -> !e.isRemoved() && (e instanceof Caster || IS_PLAYER.test(e));
Predicate IS_PLACED_SPELL = e -> e instanceof Caster && !e.isRemoved();
- Predicate IS_MAGIC_IMMUNE = e -> (e instanceof MagicImmune || !(e instanceof LivingEntity)) && !(e instanceof ItemEntity);
+ Predicate IS_MAGIC_IMMUNE = e -> (e instanceof MagicImmune || !(e instanceof LivingEntity)) && !(e instanceof ItemEntity) && !(e instanceof ExperienceOrbEntity);
Predicate EXCEPT_MAGIC_IMMUNE = IS_MAGIC_IMMUNE.negate();
Predicate VALID_LIVING_AND_NOT_MAGIC_IMMUNE = EntityPredicates.VALID_LIVING_ENTITY.and(EXCEPT_MAGIC_IMMUNE);
diff --git a/src/main/java/com/minelittlepony/unicopia/InteractionManager.java b/src/main/java/com/minelittlepony/unicopia/InteractionManager.java
index 0e507331..7c60d0d5 100644
--- a/src/main/java/com/minelittlepony/unicopia/InteractionManager.java
+++ b/src/main/java/com/minelittlepony/unicopia/InteractionManager.java
@@ -1,22 +1,19 @@
package com.minelittlepony.unicopia;
import java.util.Map;
-import java.util.Optional;
+import java.util.UUID;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
-import com.minelittlepony.unicopia.ability.magic.CasterView;
import com.minelittlepony.unicopia.entity.player.dummy.DummyPlayerEntity;
-import com.minelittlepony.unicopia.server.world.Ether;
+import com.minelittlepony.unicopia.particle.ParticleSpawner;
import com.mojang.authlib.GameProfile;
import net.minecraft.entity.Entity;
import net.minecraft.entity.player.PlayerEntity;
import net.minecraft.network.PacketByteBuf;
-import net.minecraft.server.world.ServerWorld;
import net.minecraft.util.Identifier;
-import net.minecraft.world.BlockView;
import net.minecraft.world.World;
public class InteractionManager {
@@ -37,11 +34,8 @@ public class InteractionManager {
return INSTANCE;
}
- public Optional getCasterView(BlockView view) {
- if (view instanceof ServerWorld world) {
- return Optional.of(Ether.get(world));
- }
- return Optional.empty();
+ public ParticleSpawner createBoundParticle(UUID id) {
+ return ParticleSpawner.EMPTY;
}
public Map readChapters(PacketByteBuf buf) {
@@ -92,4 +86,8 @@ public class InteractionManager {
public PlayerEntity createPlayer(World world, GameProfile profile) {
return new DummyPlayerEntity(world, profile);
}
+
+ public void sendPlayerLookAngles(PlayerEntity player) {
+
+ }
}
diff --git a/src/main/java/com/minelittlepony/unicopia/Owned.java b/src/main/java/com/minelittlepony/unicopia/Owned.java
index 37642a1d..337a59ed 100644
--- a/src/main/java/com/minelittlepony/unicopia/Owned.java
+++ b/src/main/java/com/minelittlepony/unicopia/Owned.java
@@ -5,6 +5,8 @@ import java.util.UUID;
import org.jetbrains.annotations.Nullable;
+import com.minelittlepony.unicopia.item.FriendshipBraceletItem;
+
import net.minecraft.entity.Entity;
/**
@@ -27,14 +29,23 @@ public interface Owned {
* Since {@link Owned#getMaster()} will only return if the owner is loaded, use this to perform checks
* in the owner's absence.
*/
- default Optional getMasterId() {
- return Optional.of(getMaster()).map(Entity::getUuid);
+ Optional getMasterId();
+
+ default boolean isOwnerOrFriend(Entity target) {
+ return FriendshipBraceletItem.isComrade(this, target) || isOwnerOrVehicle(target);
+ }
+
+ default boolean isOwnerOrVehicle(@Nullable Entity target) {
+ if (isOwnedBy(target)) {
+ return true;
+ }
+
+ Entity owner = getMaster();
+ return target != null && owner != null && owner.isConnectedThroughVehicle(target);
}
default boolean isOwnedBy(@Nullable Object owner) {
- return owner instanceof Entity e
- && getMasterId().isPresent()
- && e.getUuid().equals(getMasterId().get());
+ return owner instanceof Entity e && e.getUuid().equals(getMasterId().orElse(null));
}
default boolean hasCommonOwner(Owned> sibling) {
diff --git a/src/main/java/com/minelittlepony/unicopia/USounds.java b/src/main/java/com/minelittlepony/unicopia/USounds.java
index 23980e16..3f53eca3 100644
--- a/src/main/java/com/minelittlepony/unicopia/USounds.java
+++ b/src/main/java/com/minelittlepony/unicopia/USounds.java
@@ -62,11 +62,18 @@ public interface USounds {
SoundEvent ENTITY_SOMBRA_AMBIENT = register("entity.sombra.ambient");
SoundEvent ENTITY_SOMBRA_LAUGH = register("entity.sombra.laugh");
SoundEvent ENTITY_SOMBRA_SNICKER = register("entity.sombra.snicker");
- SoundEvent ENTITY_SOMBRA_SCARY = USounds.Vanilla.ENTITY_GHAST_AMBIENT;
+ SoundEvent ENTITY_SOMBRA_SCARY = ENTITY_GHAST_AMBIENT;
SoundEvent ENTITY_CRYSTAL_SHARDS_AMBIENT = BLOCK_AMETHYST_BLOCK_HIT;
SoundEvent ENTITY_CRYSTAL_SHARDS_JOSTLE = BLOCK_AMETHYST_BLOCK_BREAK;
+ SoundEvent ENTITY_IGNIMEOUS_BULB_HURT = ENTITY_WARDEN_HURT;
+ SoundEvent ENTITY_IGNIMEOUS_BULB_DEATH = ENTITY_WARDEN_DEATH;
+
+ SoundEvent ENTITY_TENTACLE_ROAR = ENTITY_RAVAGER_ROAR;
+ SoundEvent ENTITY_TENTACLE_AMBIENT = BLOCK_CONDUIT_AMBIENT_SHORT;
+ SoundEvent ENTITY_TENTACLE_DIG = ENTITY_WARDEN_DIG;
+
SoundEvent ITEM_AMULET_CHARGING = register("item.amulet.charging");
SoundEvent ITEM_AMULET_RECHARGE = register("item.amulet.recharge");
@@ -90,6 +97,8 @@ public interface USounds {
SoundEvent ITEM_STAFF_STRIKE = ENTITY_PLAYER_ATTACK_CRIT;
SoundEvent ITEM_MAGIC_STAFF_CHARGE = ENTITY_GUARDIAN_ATTACK;
+ SoundEvent ITEM_CURING_JOKE_CURE = BLOCK_AMETHYST_BLOCK_BREAK;
+
SoundEvent ITEM_ROCK_LAND = BLOCK_STONE_HIT;
RegistryEntry.Reference ITEM_MUFFIN_BOUNCE = BLOCK_NOTE_BLOCK_BANJO;
diff --git a/src/main/java/com/minelittlepony/unicopia/UTags.java b/src/main/java/com/minelittlepony/unicopia/UTags.java
index 91b4fcae..08da479c 100644
--- a/src/main/java/com/minelittlepony/unicopia/UTags.java
+++ b/src/main/java/com/minelittlepony/unicopia/UTags.java
@@ -23,6 +23,7 @@ public interface UTags {
TagKey- SHADES = item("shades");
TagKey
- CHANGELING_EDIBLE = item("food_types/changeling_edible");
TagKey
- SPOOKED_MOB_DROPS = item("spooked_mob_drops");
+ TagKey
- HAS_NO_TRAITS = item("has_no_traits");
TagKey
- IS_DELIVERED_AGGRESSIVELY = item("is_delivered_aggressively");
TagKey
- FLOATS_ON_CLOUDS = item("floats_on_clouds");
TagKey
- COOLS_OFF_KIRINS = item("cools_off_kirins");
@@ -42,6 +43,8 @@ public interface UTags {
TagKey CRYSTAL_HEART_BASE = block("crystal_heart_base");
TagKey CRYSTAL_HEART_ORNAMENT = block("crystal_heart_ornament");
+ TagKey UNAFFECTED_BY_GROW_ABILITY = block("unaffected_by_grow_ability");
+ TagKey KICKS_UP_DUST = block("kicks_up_dust");
TagKey POLEARM_MINEABLE = block("mineable/polearm");
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/AbilityDispatcher.java b/src/main/java/com/minelittlepony/unicopia/ability/AbilityDispatcher.java
index d9a977b4..8a291d0a 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/AbilityDispatcher.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/AbilityDispatcher.java
@@ -209,6 +209,7 @@ public class AbilityDispatcher implements Tickable, NbtSerialisable {
warmup = 0;
if (data.isPresent()) {
+ InteractionManager.instance().sendPlayerLookAngles(player.asEntity());
Channel.CLIENT_PLAYER_ABILITY.sendToServer(new MsgPlayerAbility<>(ability, data, ActivationType.NONE));
} else {
player.asEntity().playSound(USounds.GUI_ABILITY_FAIL, 1, 1);
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/ChangeFormAbility.java b/src/main/java/com/minelittlepony/unicopia/ability/ChangeFormAbility.java
index 4c841b1a..64fabfa4 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/ChangeFormAbility.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/ChangeFormAbility.java
@@ -76,9 +76,9 @@ public class ChangeFormAbility implements Ability {
targets.forEach(target -> {
Race supressed = target.getSuppressedRace();
if (target == player || supressed.isUnset() == isTransforming) {
- Race actualRace = target.getSpecies();
+ Race actualRace = isTransforming ? target.getSpecies() : Race.UNSET;
target.setSpecies(supressed.or(player.getCompositeRace().potential()));
- target.setSuppressedRace(isTransforming ? actualRace : Race.UNSET);
+ target.setSuppressedRace(actualRace);
}
});
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/EarthPonyGrowAbility.java b/src/main/java/com/minelittlepony/unicopia/ability/EarthPonyGrowAbility.java
index ee82de7d..61ab54d5 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/EarthPonyGrowAbility.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/EarthPonyGrowAbility.java
@@ -1,22 +1,37 @@
package com.minelittlepony.unicopia.ability;
import java.util.Optional;
-
+import java.util.function.DoubleSupplier;
+import java.util.function.Supplier;
import com.minelittlepony.unicopia.Race;
+import com.minelittlepony.unicopia.USounds;
+import com.minelittlepony.unicopia.UTags;
import com.minelittlepony.unicopia.ability.data.Hit;
import com.minelittlepony.unicopia.ability.data.Pos;
import com.minelittlepony.unicopia.block.UBlocks;
import com.minelittlepony.unicopia.entity.player.Pony;
+import com.minelittlepony.unicopia.item.TransformCropsRecipe;
+import com.minelittlepony.unicopia.item.URecipes;
import com.minelittlepony.unicopia.particle.MagicParticleEffect;
+import com.minelittlepony.unicopia.particle.ParticleUtils;
+import com.minelittlepony.unicopia.server.world.BlockDestructionManager;
import com.minelittlepony.unicopia.util.TraceHelper;
+import com.minelittlepony.unicopia.util.VecHelper;
+
+import net.minecraft.block.Block;
import net.minecraft.block.BlockState;
import net.minecraft.block.Blocks;
+import net.minecraft.block.CarrotsBlock;
+import net.minecraft.block.FarmlandBlock;
import net.minecraft.item.BoneMealItem;
import net.minecraft.item.ItemStack;
import net.minecraft.item.Items;
+import net.minecraft.particle.ParticleTypes;
import net.minecraft.util.math.BlockPos;
import net.minecraft.util.math.Direction;
+import net.minecraft.util.math.Vec3d;
import net.minecraft.world.World;
+import net.minecraft.world.WorldEvents;
/**
* Earth Pony ability to grow crops
@@ -57,10 +72,14 @@ public class EarthPonyGrowAbility implements Ability {
public boolean apply(Pony player, Pos data) {
int count = 0;
- for (BlockPos pos : BlockPos.iterate(
- data.pos().add(-2, -2, -2),
- data.pos().add( 2, 2, 2))) {
- count += applySingle(player.asWorld(), player.asWorld().getBlockState(pos), pos);
+ if (!applyDirectly(player, data.pos())) {
+ for (BlockPos pos : BlockPos.iterate(
+ data.pos().add(-2, -2, -2),
+ data.pos().add( 2, 2, 2))) {
+ count += applySingle(player, player.asWorld(), player.asWorld().getBlockState(pos), pos);
+ }
+ } else {
+ count = 1;
}
if (count > 0) {
@@ -69,7 +88,7 @@ public class EarthPonyGrowAbility implements Ability {
return true;
}
- protected int applySingle(World w, BlockState state, BlockPos pos) {
+ protected int applySingle(Pony player, World w, BlockState state, BlockPos pos) {
ItemStack stack = new ItemStack(Items.BONE_MEAL);
@@ -77,12 +96,33 @@ public class EarthPonyGrowAbility implements Ability {
return growable.grow(w, state, pos) ? 1 : 0;
}
+ if (state.isOf(Blocks.CARROTS)) {
+ if (state.get(CarrotsBlock.AGE) == CarrotsBlock.MAX_AGE) {
+ boolean transform = w.random.nextInt(3) == 0;
+ spawnConversionParticles(w, pos, transform);
+ if (transform) {
+ w.setBlockState(pos, UBlocks.GOLD_ROOT.getDefaultState().with(CarrotsBlock.AGE, CarrotsBlock.MAX_AGE));
+ }
+
+ return 5;
+ }
+ }
+
+ if (w.getBlockState(pos).isIn(UTags.UNAFFECTED_BY_GROW_ABILITY)) {
+ return 0;
+ }
+
if (BoneMealItem.useOnFertilizable(stack, w, pos)) {
if (w.random.nextInt(350) == 0) {
if (w.getBlockState(pos.down()).isOf(Blocks.FARMLAND)) {
- w.setBlockState(pos.down(), Blocks.DIRT.getDefaultState());
+ FarmlandBlock.setToDirt(null, state, w, pos.down());
}
w.setBlockState(pos, UBlocks.PLUNDER_VINE_BUD.getDefaultState());
+ } else if (w.random.nextInt(5000) == 0) {
+ if (w.getBlockState(pos.down()).isOf(Blocks.FARMLAND)) {
+ FarmlandBlock.setToDirt(null, state, w, pos.down());
+ }
+ UBlocks.CURING_JOKE.grow(w, state, pos);
}
return 1;
}
@@ -94,6 +134,56 @@ public class EarthPonyGrowAbility implements Ability {
return 0;
}
+ private boolean applyDirectly(Pony player, BlockPos pos) {
+ return player.asWorld().getRecipeManager()
+ .getAllMatches(URecipes.GROWING, new TransformCropsRecipe.PlacementArea(player, pos), player.asWorld())
+ .stream()
+ .map(recipe -> recipe.checkPattern(player.asWorld(), pos))
+ .filter(result -> result.matchedLocations().size() + 1 >= TransformCropsRecipe.MINIMUM_INPUT)
+ .filter(result -> {
+ boolean transform = result.shoudTransform(player.asWorld().random);
+
+ player.playSound(USounds.ENTITY_CRYSTAL_SHARDS_AMBIENT, 1);
+
+ result.matchedLocations().forEach(cell -> {
+ spawnConversionParticles(player.asWorld(), cell.up(), false);
+ BlockDestructionManager manager = BlockDestructionManager.of(player.asWorld());
+ if (transform) {
+ if (manager.damageBlock(cell, 8) >= BlockDestructionManager.MAX_DAMAGE || player.asWorld().random.nextInt(20) == 0) {
+ player.asWorld().setBlockState(cell, Blocks.DIRT.getDefaultState());
+ player.asWorld().syncWorldEvent(WorldEvents.BLOCK_BROKEN, cell, Block.getRawIdFromState(player.asWorld().getBlockState(cell)));
+ }
+ } else {
+ if (manager.damageBlock(cell, 4) >= BlockDestructionManager.MAX_DAMAGE || player.asWorld().random.nextInt(20) == 0) {
+ player.asWorld().setBlockState(cell, Blocks.DIRT.getDefaultState());
+ player.asWorld().syncWorldEvent(WorldEvents.BLOCK_BROKEN, cell, Block.getRawIdFromState(player.asWorld().getBlockState(cell)));
+ }
+ }
+ });
+
+ spawnConversionParticles(player.asWorld(), pos, transform);
+ if (transform) {
+ player.asWorld().setBlockState(pos, result.recipe().getResult(player.asWorld(), pos));
+ }
+
+ return true;
+ })
+ .findFirst()
+ .isPresent();
+ }
+
+ private static void spawnConversionParticles(World w, BlockPos pos, boolean success) {
+ DoubleSupplier vecComponentFactory = () -> w.random.nextTriangular(0, 0.5);
+ Supplier posSupplier = () -> pos.toCenterPos().add(VecHelper.supply(vecComponentFactory));
+
+ for (int i = 0; i < 25; i++) {
+ ParticleUtils.spawnParticle(w, new MagicParticleEffect(0xFFFF00), posSupplier.get(), Vec3d.ZERO);
+ if (success) {
+ ParticleUtils.spawnParticle(w, ParticleTypes.CLOUD, posSupplier.get(), Vec3d.ZERO);
+ }
+ }
+ }
+
@Override
public void warmUp(Pony player, AbilitySlot slot) {
player.getMagicalReserves().getExertion().addPercent(30);
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/EarthPonyStompAbility.java b/src/main/java/com/minelittlepony/unicopia/ability/EarthPonyStompAbility.java
index 1a5c72ef..e81f5a0d 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/EarthPonyStompAbility.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/EarthPonyStompAbility.java
@@ -6,6 +6,7 @@ import org.jetbrains.annotations.Nullable;
import com.minelittlepony.unicopia.AwaitTickQueue;
import com.minelittlepony.unicopia.Race;
+import com.minelittlepony.unicopia.UTags;
import com.minelittlepony.unicopia.ability.data.Hit;
import com.minelittlepony.unicopia.client.render.PlayerPoser.Animation;
import com.minelittlepony.unicopia.entity.Living;
@@ -17,6 +18,7 @@ import com.minelittlepony.unicopia.particle.ParticleUtils;
import com.minelittlepony.unicopia.particle.UParticles;
import com.minelittlepony.unicopia.server.world.BlockDestructionManager;
import com.minelittlepony.unicopia.util.PosHelper;
+import com.minelittlepony.unicopia.util.VecHelper;
import net.minecraft.block.Block;
import net.minecraft.block.BlockState;
@@ -26,6 +28,7 @@ import net.minecraft.entity.LivingEntity;
import net.minecraft.entity.attribute.EntityAttributes;
import net.minecraft.entity.player.PlayerEntity;
import net.minecraft.item.ItemStack;
+import net.minecraft.particle.BlockStateParticleEffect;
import net.minecraft.registry.tag.BlockTags;
import net.minecraft.server.world.ServerWorld;
import net.minecraft.util.Identifier;
@@ -33,6 +36,7 @@ import net.minecraft.util.math.BlockBox;
import net.minecraft.util.math.BlockPos;
import net.minecraft.util.math.Box;
import net.minecraft.util.math.MathHelper;
+import net.minecraft.util.math.Vec3d;
import net.minecraft.world.World;
import net.minecraft.world.WorldEvents;
@@ -156,7 +160,10 @@ public class EarthPonyStompAbility implements Ability {
spawnEffectAround(player, center, radius, rad);
ParticleUtils.spawnParticle(player.getWorld(), UParticles.GROUND_POUND, player.getX(), player.getY() - 1, player.getZ(), 0, 0, 0);
- ParticleUtils.spawnParticle(player.getWorld(), UParticles.SHOCKWAVE, player.getX(), player.getY() - 1, player.getZ(), 0, 0, 0);
+ BlockState steppingState = player.getSteppingBlockState();
+ if (steppingState.isIn(UTags.KICKS_UP_DUST)) {
+ ParticleUtils.spawnParticle(player.getWorld(), new BlockStateParticleEffect(UParticles.DUST_CLOUD, steppingState), player.getBlockPos().down().toCenterPos(), Vec3d.ZERO);
+ }
iplayer.subtractEnergyCost(rad);
iplayer.asEntity().addExhaustion(3);
@@ -218,6 +225,12 @@ public class EarthPonyStompAbility implements Ability {
} else {
w.syncWorldEvent(WorldEvents.BLOCK_BROKEN, pos, Block.getRawIdFromState(state));
}
+
+ if (state.isIn(UTags.KICKS_UP_DUST)) {
+ if (w.random.nextInt(4) == 0 && w.isAir(pos.up()) && w.getFluidState(pos.up()).isEmpty()) {
+ ParticleUtils.spawnParticle(w, new BlockStateParticleEffect(UParticles.DUST_CLOUD, state), pos.up().toCenterPos(), VecHelper.supply(() -> w.random.nextTriangular(0, 0.1F)));
+ }
+ }
}
@Override
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/PegasusRainboomAbility.java b/src/main/java/com/minelittlepony/unicopia/ability/PegasusRainboomAbility.java
index 087e1cb0..bbe7256e 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/PegasusRainboomAbility.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/PegasusRainboomAbility.java
@@ -10,10 +10,6 @@ import com.minelittlepony.unicopia.ability.magic.spell.CastingMethod;
import com.minelittlepony.unicopia.ability.magic.spell.effect.SpellType;
import com.minelittlepony.unicopia.entity.player.Pony;
import com.minelittlepony.unicopia.particle.MagicParticleEffect;
-import com.minelittlepony.unicopia.particle.OrientedBillboardParticleEffect;
-import com.minelittlepony.unicopia.particle.UParticles;
-
-import net.minecraft.util.math.Vec3d;
/**
* Pegasus ability to perform rainbooms
@@ -72,7 +68,6 @@ public class PegasusRainboomAbility implements Ability {
}
if (player.consumeSuperMove()) {
- player.addParticle(new OrientedBillboardParticleEffect(UParticles.RAINBOOM_RING, player.getPhysics().getMotionAngle()), player.getOriginVector(), Vec3d.ZERO);
SpellType.RAINBOOM.withTraits().apply(player, CastingMethod.INNATE);
}
return true;
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/UnicornDispellAbility.java b/src/main/java/com/minelittlepony/unicopia/ability/UnicornDispellAbility.java
index 99dda711..d965f9da 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/UnicornDispellAbility.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/UnicornDispellAbility.java
@@ -7,6 +7,7 @@ import com.minelittlepony.unicopia.InteractionManager;
import com.minelittlepony.unicopia.Race;
import com.minelittlepony.unicopia.ability.data.Pos;
import com.minelittlepony.unicopia.ability.magic.Caster;
+import com.minelittlepony.unicopia.ability.magic.SpellContainer.Operation;
import com.minelittlepony.unicopia.ability.magic.spell.effect.SpellType;
import com.minelittlepony.unicopia.client.render.PlayerPoser.Animation;
import com.minelittlepony.unicopia.entity.player.Pony;
@@ -92,7 +93,11 @@ public class UnicornDispellAbility implements Ability {
public boolean apply(Pony player, Pos data) {
player.setAnimation(Animation.WOLOLO, Animation.Recipient.ANYONE);
Caster.stream(VecHelper.findInRange(player.asEntity(), player.asWorld(), data.vec(), 3, EquinePredicates.IS_PLACED_SPELL).stream()).forEach(target -> {
- target.getSpellSlot().clear();
+ target.getSpellSlot().forEach(spell -> {
+ spell.setDead();
+ spell.tickDying(target);
+ return Operation.ofBoolean(!spell.isDead());
+ }, true);
});
return true;
}
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/Affine.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/Affine.java
index 007da627..7cdfdaef 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/Affine.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/Affine.java
@@ -19,4 +19,8 @@ public interface Affine {
default boolean isFriendlyTogether(Affine other) {
return getAffinity() != Affinity.BAD && other.getAffinity() != Affinity.BAD;
}
+
+ default boolean applyInversion(Affine other, boolean friendly) {
+ return isEnemy(other) ? !friendly : friendly;
+ }
}
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/Caster.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/Caster.java
index 98264836..150b1004 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/Caster.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/Caster.java
@@ -11,6 +11,7 @@ import com.minelittlepony.unicopia.ability.magic.spell.effect.SpellType;
import com.minelittlepony.unicopia.entity.*;
import com.minelittlepony.unicopia.entity.damage.UDamageSources;
import com.minelittlepony.unicopia.particle.ParticleSource;
+import com.minelittlepony.unicopia.server.world.Ether;
import com.minelittlepony.unicopia.server.world.ModificationType;
import com.minelittlepony.unicopia.util.SoundEmitter;
import com.minelittlepony.unicopia.util.VecHelper;
@@ -99,11 +100,7 @@ public interface Caster extends
}
default boolean canCastAt(Vec3d pos) {
- return findAllSpellsInRange(500, SpellType.ARCANE_PROTECTION::isOn).noneMatch(caster -> caster
- .getSpellSlot().get(SpellType.ARCANE_PROTECTION, false)
- .filter(spell -> spell.blocksMagicFor(caster, this, pos))
- .isPresent()
- );
+ return !Ether.get(asWorld()).anyMatch(SpellType.ARCANE_PROTECTION, (spell, caster) -> spell.blocksMagicFor(caster, this, pos));
}
static Stream> stream(Stream entities) {
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/CasterView.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/CasterView.java
deleted file mode 100644
index 572f4542..00000000
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/CasterView.java
+++ /dev/null
@@ -1,44 +0,0 @@
-package com.minelittlepony.unicopia.ability.magic;
-
-import java.util.Map;
-import java.util.function.Predicate;
-import java.util.stream.Stream;
-
-import org.jetbrains.annotations.Nullable;
-
-import com.minelittlepony.unicopia.EquinePredicates;
-import com.minelittlepony.unicopia.ability.magic.spell.Spell;
-import com.minelittlepony.unicopia.util.VecHelper;
-
-import net.minecraft.entity.Entity;
-import net.minecraft.util.math.BlockPos;
-import net.minecraft.util.math.Vec3d;
-import net.minecraft.world.*;
-
-public interface CasterView {
- EntityView getWorld();
-
- default
Stream, S>> findAllSpellsInRange(BlockPos pos, double radius, SpellPredicate type) {
- return findAllCastersInRange(pos, radius).flatMap(caster -> {
- return caster.getSpellSlot().stream(type, false).map(spell -> {
- return Map.entry(caster, spell);
- });
- });
- }
-
- default Stream> findAllCastersInRange(BlockPos pos, double radius) {
- return findAllCastersInRange(pos, radius, null);
- }
-
- default Stream> findAllCastersInRange(BlockPos pos, double radius, @Nullable Predicate test) {
- return Caster.stream(findAllEntitiesInRange(pos, radius, test == null ? EquinePredicates.IS_CASTER : EquinePredicates.IS_CASTER.and(test)));
- }
-
- default Stream findAllEntitiesInRange(BlockPos pos, double radius, @Nullable Predicate test) {
- return VecHelper.findInRange(null, getWorld(), Vec3d.ofCenter(pos), radius, test).stream();
- }
-
- default Stream findAllEntitiesInRange(BlockPos pos, double radius) {
- return findAllEntitiesInRange(pos, radius, null);
- }
-}
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/SpellContainer.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/SpellContainer.java
index bbe1ad70..e703911d 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/SpellContainer.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/SpellContainer.java
@@ -39,6 +39,13 @@ public interface SpellContainer {
*/
void put(@Nullable Spell effect);
+ /**
+ * Cleanly removes a spell from this spell container.
+ *
+ * @param spellid ID of the spell to remove.
+ */
+ void remove(UUID spellid);
+
/**
* Removes all active effects that match or contain a matching effect.
*
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/AbstractDelegatingSpell.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/AbstractDelegatingSpell.java
index f6219d62..68a006ed 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/AbstractDelegatingSpell.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/AbstractDelegatingSpell.java
@@ -69,11 +69,20 @@ public abstract class AbstractDelegatingSpell implements Spell,
getDelegates().forEach(Spell::setDead);
}
+ @Override
+ public void tickDying(Caster> caster) {
+ }
+
@Override
public boolean isDead() {
return getDelegates().isEmpty() || getDelegates().stream().allMatch(Spell::isDead);
}
+ @Override
+ public boolean isDying() {
+ return false;
+ }
+
@Override
public boolean isDirty() {
return dirty || getDelegates().stream().anyMatch(Spell::isDirty);
@@ -110,7 +119,13 @@ public abstract class AbstractDelegatingSpell implements Spell,
@Override
public boolean tick(Caster> source, Situation situation) {
- return execute(getDelegates().stream(), a -> a.tick(source, situation));
+ return execute(getDelegates().stream(), a -> {
+ if (a.isDying()) {
+ a.tickDying(source);
+ return !a.isDead();
+ }
+ return a.tick(source, situation);
+ });
}
@Override
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/PlaceableSpell.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/PlaceableSpell.java
index 7154056a..ed03506d 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/PlaceableSpell.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/PlaceableSpell.java
@@ -13,10 +13,6 @@ import com.minelittlepony.unicopia.entity.mob.UEntities;
import com.minelittlepony.unicopia.entity.player.Pony;
import com.minelittlepony.unicopia.network.Channel;
import com.minelittlepony.unicopia.network.MsgCasterLookRequest;
-import com.minelittlepony.unicopia.particle.OrientedBillboardParticleEffect;
-import com.minelittlepony.unicopia.particle.ParticleHandle;
-import com.minelittlepony.unicopia.particle.UParticles;
-import com.minelittlepony.unicopia.particle.ParticleHandle.Attachment;
import com.minelittlepony.unicopia.server.world.Ether;
import com.minelittlepony.unicopia.util.NbtSerialisable;
@@ -24,6 +20,7 @@ import net.minecraft.nbt.*;
import net.minecraft.registry.*;
import net.minecraft.server.network.ServerPlayerEntity;
import net.minecraft.util.Identifier;
+import net.minecraft.util.math.MathHelper;
import net.minecraft.util.math.Vec3d;
import net.minecraft.world.World;
@@ -41,9 +38,10 @@ public class PlaceableSpell extends AbstractDelegatingSpell implements OrientedS
private RegistryKey dimension;
/**
- * The visual effect
+ * ID of the placed counterpart of this spell.
*/
- private final ParticleHandle particlEffect = new ParticleHandle();
+ @Nullable
+ private UUID placedSpellId;
/**
* The cast spell entity
@@ -58,6 +56,13 @@ public class PlaceableSpell extends AbstractDelegatingSpell implements OrientedS
public float pitch;
public float yaw;
+ private int prevAge;
+ private int age;
+
+ private boolean dead;
+ private int prevDeathTicks;
+ private int deathTicks;
+
private Optional position = Optional.empty();
public PlaceableSpell(CustomisedSpellType> type) {
@@ -69,23 +74,42 @@ public class PlaceableSpell extends AbstractDelegatingSpell implements OrientedS
return this;
}
+ public float getAge(float tickDelta) {
+ return MathHelper.lerp(tickDelta, prevAge, age);
+ }
+
+ public float getScale(float tickDelta) {
+ float add = MathHelper.clamp(getAge(tickDelta) / 25F, 0, 1);
+ float subtract = dead ? 1 - (MathHelper.lerp(tickDelta, prevDeathTicks, deathTicks) / 20F) : 0;
+ return MathHelper.clamp(add - subtract, 0, 1);
+ }
+
+ @Override
+ public boolean isDying() {
+ return dead && deathTicks > 0;
+ }
+
+ @Override
+ public void setDead() {
+ super.setDead();
+ dead = true;
+ deathTicks = 20;
+ }
+
+ @Override
+ public boolean isDead() {
+ return dead && deathTicks <= 0;
+ }
+
@Override
public Collection getDelegates() {
return List.of(spell);
}
- @Override
- public void setDead() {
- super.setDead();
- particlEffect.destroy();
- }
-
@Override
public boolean tick(Caster> source, Situation situation) {
-
if (situation == Situation.BODY) {
if (!source.isClient()) {
-
if (dimension == null) {
dimension = source.asWorld().getRegistryKey();
if (source instanceof Pony) {
@@ -105,29 +129,31 @@ public class PlaceableSpell extends AbstractDelegatingSpell implements OrientedS
if (situation == Situation.GROUND_ENTITY) {
if (!source.isClient()) {
- Ether ether = Ether.get(source.asWorld());
- if (ether.getEntry(getType(), source).isEmpty()) {
+ if (Ether.get(source.asWorld()).get(this, source) == null) {
setDead();
return false;
}
}
- if (spell instanceof PlacementDelegate delegate) {
- delegate.updatePlacement(source, this);
+ prevAge = age;
+ if (age < 25) {
+ age++;
}
- getParticleEffectAttachment(source).ifPresent(p -> {
- p.setAttribute(Attachment.ATTR_COLOR, spell.getType().getColor());
- });
-
return super.tick(source, Situation.GROUND);
}
return !isDead();
}
+ @Override
+ public void tickDying(Caster> caster) {
+ prevDeathTicks = deathTicks;
+ deathTicks--;
+ }
+
private void checkDetachment(Caster> source, EntityValues> target) {
- if (getWorld(source).map(Ether::get).flatMap(ether -> ether.getEntry(getType(), target.uuid())).isEmpty()) {
+ if (getWorld(source).map(Ether::get).map(ether -> ether.get(getType(), target, placedSpellId)).isEmpty()) {
setDead();
}
}
@@ -143,7 +169,8 @@ public class PlaceableSpell extends AbstractDelegatingSpell implements OrientedS
entity.getSpellSlot().put(copy);
entity.setCaster(source);
entity.getWorld().spawnEntity(entity);
- Ether.get(entity.getWorld()).put(getType(), entity);
+ placedSpellId = copy.getUuid();
+ Ether.get(entity.getWorld()).getOrCreate(copy, entity);
castEntity.set(entity);
setDirty();
@@ -174,8 +201,7 @@ public class PlaceableSpell extends AbstractDelegatingSpell implements OrientedS
if (!source.isClient()) {
castEntity.getTarget().ifPresent(target -> {
getWorld(source).map(Ether::get)
- .flatMap(ether -> ether.getEntry(getType(), target.uuid()))
- .ifPresent(Ether.Entry::markDead);
+ .ifPresent(ether -> ether.remove(getType(), target.uuid()));
});
castEntity.set(null);
getSpellEntity(source).ifPresent(e -> {
@@ -183,7 +209,7 @@ public class PlaceableSpell extends AbstractDelegatingSpell implements OrientedS
});
if (source.asEntity() instanceof CastSpellEntity spellcast) {
- Ether.get(source.asWorld()).remove(getType(), source);
+ Ether.get(source.asWorld()).remove(this, source);
}
}
super.onDestroyed(source);
@@ -197,12 +223,6 @@ public class PlaceableSpell extends AbstractDelegatingSpell implements OrientedS
return castEntity.getTarget().map(EntityValues::pos);
}
- public Optional getParticleEffectAttachment(Caster> source) {
- return particlEffect.update(getUuid(), source, spawner -> {
- spawner.addParticle(new OrientedBillboardParticleEffect(UParticles.MAGIC_RUNES, pitch + 90, yaw), Vec3d.ZERO, Vec3d.ZERO);
- });
- }
-
protected Optional getWorld(Caster> source) {
return Optional.ofNullable(dimension)
.map(dim -> source.asWorld().getServer().getWorld(dim));
@@ -211,11 +231,17 @@ public class PlaceableSpell extends AbstractDelegatingSpell implements OrientedS
@Override
public void toNBT(NbtCompound compound) {
super.toNBT(compound);
+ compound.putBoolean("dead", dead);
+ compound.putInt("deathTicks", deathTicks);
+ compound.putInt("age", age);
compound.putFloat("pitch", pitch);
compound.putFloat("yaw", yaw);
position.ifPresent(pos -> {
compound.put("position", NbtSerialisable.writeVector(pos));
});
+ if (placedSpellId != null) {
+ compound.putUuid("placedSpellId", placedSpellId);
+ }
if (dimension != null) {
compound.putString("dimension", dimension.getValue().toString());
}
@@ -226,9 +252,13 @@ public class PlaceableSpell extends AbstractDelegatingSpell implements OrientedS
@Override
public void fromNBT(NbtCompound compound) {
super.fromNBT(compound);
+ dead = compound.getBoolean("dead");
+ deathTicks = compound.getInt("deathTicks");
+ age = compound.getInt("age");
pitch = compound.getFloat("pitch");
yaw = compound.getFloat("yaw");
position = compound.contains("position") ? Optional.of(NbtSerialisable.readVector(compound.getList("position", NbtElement.FLOAT_TYPE))) : Optional.empty();
+ placedSpellId = compound.containsUuid("placedSpellId") ? compound.getUuid("placedSpellId") : null;
if (compound.contains("dimension", NbtElement.STRING_TYPE)) {
Identifier id = Identifier.tryParse(compound.getString("dimension"));
if (id != null) {
@@ -257,9 +287,6 @@ public class PlaceableSpell extends AbstractDelegatingSpell implements OrientedS
}
public interface PlacementDelegate {
-
void onPlaced(Caster> source, PlaceableSpell parent, CastSpellEntity entity);
-
- void updatePlacement(Caster> source, PlaceableSpell parent);
}
}
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/RainboomAbilitySpell.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/RainboomAbilitySpell.java
index 4bf0d726..6c6d1099 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/RainboomAbilitySpell.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/RainboomAbilitySpell.java
@@ -1,12 +1,16 @@
package com.minelittlepony.unicopia.ability.magic.spell;
+import org.jetbrains.annotations.Nullable;
+
+import com.minelittlepony.unicopia.InteractionManager;
import com.minelittlepony.unicopia.UTags;
import com.minelittlepony.unicopia.ability.magic.Caster;
import com.minelittlepony.unicopia.ability.magic.spell.effect.*;
import com.minelittlepony.unicopia.entity.damage.UDamageTypes;
import com.minelittlepony.unicopia.entity.player.Pony;
-import com.minelittlepony.unicopia.particle.ParticleHandle;
-import com.minelittlepony.unicopia.particle.ParticleHandle.Attachment;
+import com.minelittlepony.unicopia.particle.OrientedBillboardParticleEffect;
+import com.minelittlepony.unicopia.particle.ParticleSpawner;
+import com.minelittlepony.unicopia.particle.TargetBoundParticleEffect;
import com.minelittlepony.unicopia.server.world.ModificationType;
import com.minelittlepony.unicopia.particle.UParticles;
import com.minelittlepony.unicopia.util.shape.Shape;
@@ -26,7 +30,8 @@ public class RainboomAbilitySpell extends AbstractSpell {
private static final int RADIUS = 5;
private static final Shape EFFECT_RANGE = new Sphere(false, RADIUS);
- private final ParticleHandle particlEffect = new ParticleHandle();
+ @Nullable
+ private ParticleSpawner boundParticle;
private int age;
@@ -35,11 +40,6 @@ public class RainboomAbilitySpell extends AbstractSpell {
setHidden(true);
}
- @Override
- protected void onDestroyed(Caster> source) {
- particlEffect.destroy();
- }
-
@Override
public boolean tick(Caster> source, Situation situation) {
@@ -47,14 +47,15 @@ public class RainboomAbilitySpell extends AbstractSpell {
return false;
}
- particlEffect.update(getUuid(), source, spawner -> {
- spawner.addParticle(UParticles.RAINBOOM_TRAIL, source.getOriginVector(), Vec3d.ZERO);
- }).ifPresent(attachment -> {
- attachment.setAttribute(Attachment.ATTR_BOUND, 1);
- });
-
if (source.isClient()) {
- // source.addParticle(new OrientedBillboardParticleEffect(UParticles.RAINBOOM_RING, source.getPhysics().getMotionAngle()), source.getOriginVector(), Vec3d.ZERO);
+ if (boundParticle == null) {
+ boundParticle = InteractionManager.INSTANCE.createBoundParticle(getUuid());
+ }
+ boundParticle.addParticle(new TargetBoundParticleEffect(UParticles.RAINBOOM_TRAIL, source.asEntity()), source.getOriginVector(), Vec3d.ZERO);
+
+ if (age == 0) {
+ source.addParticle(new OrientedBillboardParticleEffect(UParticles.RAINBOOM_RING, source.getPhysics().getMotionAngle()), source.getOriginVector(), Vec3d.ZERO);
+ }
}
source.findAllEntitiesInRange(RADIUS).forEach(e -> {
@@ -92,5 +93,6 @@ public class RainboomAbilitySpell extends AbstractSpell {
public void fromNBT(NbtCompound compound) {
super.fromNBT(compound);
age = compound.getInt("age");
+ boundParticle = null;
}
}
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/Spell.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/Spell.java
index d6bb1ccf..55a5f3bf 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/Spell.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/Spell.java
@@ -61,6 +61,8 @@ public interface Spell extends NbtSerialisable, Affine {
*/
boolean isDead();
+ boolean isDying();
+
/**
* Returns true if this effect has changes that need to be sent to the client.
*/
@@ -68,6 +70,7 @@ public interface Spell extends NbtSerialisable, Affine {
/**
* Applies this spell to the supplied caster.
+ * @param caster The caster to apply the spell to
*/
default boolean apply(Caster> caster) {
caster.getSpellSlot().put(this);
@@ -76,7 +79,7 @@ public interface Spell extends NbtSerialisable, Affine {
/**
* Gets the default form of this spell used to apply to a caster.
- * @param caster
+ * @param caster The caster currently fueling this spell
*/
default Spell prepareForCast(Caster> caster, CastingMethod method) {
return this;
@@ -89,6 +92,12 @@ public interface Spell extends NbtSerialisable, Affine {
*/
boolean tick(Caster> caster, Situation situation);
+ /**
+ * Called on spells that are actively dying to update any post-death animations before removal.
+ * @param caster The caster currently fueling this spell
+ */
+ void tickDying(Caster> caster);
+
/**
* Marks this effect as dirty.
*/
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/TimedSpell.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/TimedSpell.java
index 9e19b8ae..d81e4f39 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/TimedSpell.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/TimedSpell.java
@@ -29,7 +29,7 @@ public interface TimedSpell extends Spell {
}
public float getPercentTimeRemaining(float tickDelta) {
- return MathHelper.lerp(tickDelta, prevDuration, duration) / maxDuration;
+ return MathHelper.lerp(tickDelta, prevDuration, duration) / (float)maxDuration;
}
public int getTicksRemaining() {
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/crafting/AltarRecipeMatch.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/crafting/AltarRecipeMatch.java
new file mode 100644
index 00000000..820f7667
--- /dev/null
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/crafting/AltarRecipeMatch.java
@@ -0,0 +1,42 @@
+package com.minelittlepony.unicopia.ability.magic.spell.crafting;
+
+import java.util.List;
+
+import org.jetbrains.annotations.Nullable;
+
+import com.minelittlepony.unicopia.item.UItems;
+
+import net.minecraft.entity.Entity;
+import net.minecraft.entity.ItemEntity;
+import net.minecraft.item.ItemStack;
+import net.minecraft.item.Items;
+
+public record AltarRecipeMatch(
+ ItemEntity target,
+ List ingredients,
+ ItemStack result
+ ) {
+
+ @Nullable
+ public static AltarRecipeMatch of(List inputs) {
+ ItemEntity clock = inputs.stream().filter(item -> item.getStack().isOf(Items.CLOCK)).findFirst().orElse(null);
+
+ if (clock != null) {
+ return new AltarRecipeMatch(clock, List.of(), UItems.SPECTRAL_CLOCK.getDefaultStack());
+ }
+
+ return null;
+ }
+
+ public boolean isRemoved() {
+ return target.isRemoved() || ingredients.stream().anyMatch(ItemEntity::isRemoved);
+ }
+
+ public void craft() {
+ ItemStack clockStack = result.copyWithCount(target.getStack().getCount());
+ clockStack.setNbt(target.getStack().getNbt());
+ target.setStack(clockStack);
+ target.setInvulnerable(true);
+ ingredients.forEach(Entity::discard);
+ }
+}
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/AbstractSpell.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/AbstractSpell.java
index a8c1b706..8b64d9c3 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/AbstractSpell.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/AbstractSpell.java
@@ -12,6 +12,7 @@ import net.minecraft.nbt.NbtCompound;
public abstract class AbstractSpell implements Spell {
private boolean dead;
+ private boolean dying;
private boolean dirty;
private boolean hidden;
private boolean destroyed;
@@ -45,7 +46,7 @@ public abstract class AbstractSpell implements Spell {
@Override
public final void setDead() {
- dead = true;
+ dying = true;
setDirty();
}
@@ -54,6 +55,11 @@ public abstract class AbstractSpell implements Spell {
return dead;
}
+ @Override
+ public final boolean isDying() {
+ return dying;
+ }
+
@Override
public final boolean isDirty() {
return dirty;
@@ -82,6 +88,11 @@ public abstract class AbstractSpell implements Spell {
protected void onDestroyed(Caster> caster) {
}
+ @Override
+ public void tickDying(Caster> caster) {
+ dead = true;
+ }
+
@Override
public final void destroy(Caster> caster) {
if (destroyed) {
@@ -94,6 +105,7 @@ public abstract class AbstractSpell implements Spell {
@Override
public void toNBT(NbtCompound compound) {
+ compound.putBoolean("dying", dying);
compound.putBoolean("dead", dead);
compound.putBoolean("hidden", hidden);
compound.putUuid("uuid", uuid);
@@ -106,6 +118,7 @@ public abstract class AbstractSpell implements Spell {
if (compound.contains("uuid")) {
uuid = compound.getUuid("uuid");
}
+ dying = compound.getBoolean("dying");
dead = compound.getBoolean("dead");
hidden = compound.getBoolean("hidden");
if (compound.contains("traits")) {
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/AreaProtectionSpell.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/AreaProtectionSpell.java
index 75615785..920c5bd5 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/AreaProtectionSpell.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/AreaProtectionSpell.java
@@ -9,6 +9,7 @@ import com.minelittlepony.unicopia.entity.mob.UEntities;
import com.minelittlepony.unicopia.entity.player.Pony;
import com.minelittlepony.unicopia.item.FriendshipBraceletItem;
import com.minelittlepony.unicopia.particle.MagicParticleEffect;
+import com.minelittlepony.unicopia.server.world.Ether;
import com.minelittlepony.unicopia.util.shape.Sphere;
import net.minecraft.entity.Entity;
@@ -42,6 +43,8 @@ public class AreaProtectionSpell extends AbstractAreaEffectSpell {
source.addParticle(new MagicParticleEffect(getType().getColor()), pos, Vec3d.ZERO);
}
});
+ } else {
+ Ether.get(source.asWorld()).getOrCreate(this, source);
}
source.findAllSpellsInRange(radius, e -> isValidTarget(source, e)).filter(caster -> !caster.hasCommonOwner(source)).forEach(caster -> {
@@ -51,6 +54,11 @@ public class AreaProtectionSpell extends AbstractAreaEffectSpell {
return !isDead();
}
+ @Override
+ protected void onDestroyed(Caster> caster) {
+ Ether.get(caster.asWorld()).remove(this, caster);
+ }
+
/**
* Calculates the maximum radius of the shield. aka The area of effect.
*/
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/AttractiveSpell.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/AttractiveSpell.java
index 669411a1..9af5cddb 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/AttractiveSpell.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/AttractiveSpell.java
@@ -7,7 +7,6 @@ import com.minelittlepony.unicopia.entity.EntityReference;
import com.minelittlepony.unicopia.entity.Living;
import com.minelittlepony.unicopia.entity.damage.UDamageTypes;
import com.minelittlepony.unicopia.particle.FollowingParticleEffect;
-import com.minelittlepony.unicopia.particle.MagicParticleEffect;
import com.minelittlepony.unicopia.particle.UParticles;
import com.minelittlepony.unicopia.projectile.MagicProjectileEntity;
import com.minelittlepony.unicopia.projectile.ProjectileDelegate;
@@ -16,6 +15,7 @@ import com.minelittlepony.unicopia.util.shape.Sphere;
import net.minecraft.entity.Entity;
import net.minecraft.entity.ItemEntity;
import net.minecraft.nbt.NbtCompound;
+import net.minecraft.particle.ParticleTypes;
import net.minecraft.util.hit.EntityHitResult;
import net.minecraft.util.math.MathHelper;
import net.minecraft.util.math.Vec3d;
@@ -44,9 +44,10 @@ public class AttractiveSpell extends ShieldSpell implements HomingSpell, TimedSp
if (timer.getTicksRemaining() <= 0) {
return false;
}
+
+ setDirty();
}
- setDirty();
target.getOrEmpty(caster.asWorld())
.filter(entity -> entity.distanceTo(caster.asEntity()) > getDrawDropOffRange(caster))
.ifPresent(entity -> {
@@ -59,12 +60,13 @@ public class AttractiveSpell extends ShieldSpell implements HomingSpell, TimedSp
@Override
public void generateParticles(Caster> source) {
- double range = getDrawDropOffRange(source) + 10;
+ double range = getDrawDropOffRange(source);
+ Vec3d origin = getOrigin(source);
- source.spawnParticles(getOrigin(source), new Sphere(false, range), 7, p -> {
+ source.spawnParticles(origin, new Sphere(false, range), 7, p -> {
source.addParticle(
- new FollowingParticleEffect(UParticles.HEALTH_DRAIN, source.asEntity(), 0.4F)
- .withChild(new MagicParticleEffect(getType().getColor())),
+ new FollowingParticleEffect(UParticles.HEALTH_DRAIN, origin, 0.4F)
+ .withChild(ParticleTypes.AMBIENT_ENTITY_EFFECT),
p,
Vec3d.ZERO
);
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/BubbleSpell.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/BubbleSpell.java
index c2b837b0..63afe303 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/BubbleSpell.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/BubbleSpell.java
@@ -11,12 +11,10 @@ import com.minelittlepony.unicopia.ability.magic.spell.trait.Trait;
import com.minelittlepony.unicopia.entity.*;
import com.minelittlepony.unicopia.entity.mob.UEntityAttributes;
import com.minelittlepony.unicopia.entity.player.Pony;
-import com.minelittlepony.unicopia.particle.ParticleHandle;
-import com.minelittlepony.unicopia.particle.SphereParticleEffect;
import com.minelittlepony.unicopia.particle.UParticles;
-import com.minelittlepony.unicopia.particle.ParticleHandle.Attachment;
import com.minelittlepony.unicopia.projectile.MagicProjectileEntity;
import com.minelittlepony.unicopia.projectile.ProjectileDelegate;
+import com.minelittlepony.unicopia.util.shape.Sphere;
import net.minecraft.entity.Entity;
import net.minecraft.entity.LivingEntity;
@@ -48,11 +46,11 @@ public class BubbleSpell extends AbstractSpell implements TimedSpell,
.with(Trait.POWER, 1)
.build();
- protected final ParticleHandle particlEffect = new ParticleHandle();
-
private final Timer timer;
private int struggles;
+
+ private float prevRadius;
private float radius;
protected BubbleSpell(CustomisedSpellType> type) {
@@ -66,6 +64,10 @@ public class BubbleSpell extends AbstractSpell implements TimedSpell,
return timer;
}
+ public float getRadius(float tickDelta) {
+ return MathHelper.lerp(tickDelta, prevRadius, radius);
+ }
+
@Override
public boolean apply(Caster> source) {
@@ -95,14 +97,19 @@ public class BubbleSpell extends AbstractSpell implements TimedSpell,
public boolean tick(Caster> source, Situation situation) {
if (situation == Situation.PROJECTILE) {
-
- source.spawnParticles(ParticleTypes.BUBBLE, 2);
+ source.spawnParticles(UParticles.BUBBLE, 2);
return true;
}
timer.tick();
- if (timer.getTicksRemaining() <= 0) {
+ boolean done = timer.getTicksRemaining() <= 0;
+
+ source.spawnParticles(source.getOriginVector().add(0, 1, 0), new Sphere(true, radius * (done ? 0.25F : 0.5F)), done ? 13 : 1, pos -> {
+ source.addParticle(done ? ParticleTypes.BUBBLE_POP : UParticles.BUBBLE, pos, Vec3d.ZERO);
+ });
+
+ if (done) {
return false;
}
@@ -116,7 +123,7 @@ public class BubbleSpell extends AbstractSpell implements TimedSpell,
source.asEntity().fallDistance = 0;
- Vec3d origin = source.getOriginVector();
+ prevRadius = radius;
if (source instanceof Pony pony && pony.sneakingChanged() && pony.asEntity().isSneaking()) {
setDirty();
@@ -128,18 +135,11 @@ public class BubbleSpell extends AbstractSpell implements TimedSpell,
}
}
- particlEffect.update(getUuid(), source, spawner -> {
- spawner.addParticle(new SphereParticleEffect(UParticles.SPHERE, 0xFFFFFF, 0.3F, 0, new Vec3d(0, radius / 2F, 0)), origin, Vec3d.ZERO);
- }).ifPresent(p -> {
- p.setAttribute(Attachment.ATTR_RADIUS, radius);
- });
-
return !isDead();
}
@Override
protected void onDestroyed(Caster> source) {
- particlEffect.destroy();
if (source.asEntity() instanceof LivingEntity l) {
MODIFIERS.forEach((attribute, modifier) -> {
if (l.getAttributes().hasAttribute(attribute)) {
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/DarkVortexSpell.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/DarkVortexSpell.java
index bda10118..02eb96d3 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/DarkVortexSpell.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/DarkVortexSpell.java
@@ -10,11 +10,11 @@ import com.minelittlepony.unicopia.ability.magic.spell.Situation;
import com.minelittlepony.unicopia.ability.magic.spell.Spell;
import com.minelittlepony.unicopia.ability.magic.spell.trait.SpellTraits;
import com.minelittlepony.unicopia.ability.magic.spell.trait.Trait;
+import com.minelittlepony.unicopia.entity.Living;
import com.minelittlepony.unicopia.entity.damage.UDamageTypes;
-import com.minelittlepony.unicopia.particle.ParticleHandle.Attachment;
+import com.minelittlepony.unicopia.particle.FollowingParticleEffect;
import com.minelittlepony.unicopia.particle.LightningBoltParticleEffect;
import com.minelittlepony.unicopia.particle.ParticleUtils;
-import com.minelittlepony.unicopia.particle.SphereParticleEffect;
import com.minelittlepony.unicopia.particle.UParticles;
import com.minelittlepony.unicopia.projectile.MagicProjectileEntity;
import com.minelittlepony.unicopia.projectile.ProjectileDelegate;
@@ -37,10 +37,6 @@ import net.minecraft.world.World.ExplosionSourceType;
/**
* More powerful version of the vortex spell which creates a black hole.
- *
- * TODO: Possible uses
- * - Garbage bin
- * - Link with a teleportation spell to create a wormhole
*/
public class DarkVortexSpell extends AttractiveSpell implements ProjectileDelegate.BlockHitListener {
public static final SpellTraits DEFAULT_TRAITS = new SpellTraits.Builder()
@@ -52,7 +48,6 @@ public class DarkVortexSpell extends AttractiveSpell implements ProjectileDelega
private static final Vec3d SPHERE_OFFSET = new Vec3d(0, 2, 0);
- private int age = 0;
private float accumulatedMass = 0;
protected DarkVortexSpell(CustomisedSpellType> type) {
@@ -84,17 +79,10 @@ public class DarkVortexSpell extends AttractiveSpell implements ProjectileDelega
return true;
}
- age++;
- setDirty();
-
- if (age % 20 == 0) {
+ if (source.asEntity().age % 20 == 0) {
source.asWorld().playSound(null, source.getOrigin(), USounds.AMBIENT_DARK_VORTEX_ADDITIONS, SoundCategory.AMBIENT, 1, 1);
}
- if (!source.subtractEnergyCost(-accumulatedMass)) {
- setDead();
- }
-
if (!source.isClient() && source.asWorld().random.nextInt(300) == 0) {
ParticleUtils.spawnParticle(source.asWorld(), LightningBoltParticleEffect.DEFAULT, getOrigin(source), Vec3d.ZERO);
}
@@ -102,6 +90,13 @@ public class DarkVortexSpell extends AttractiveSpell implements ProjectileDelega
return super.tick(source, situation);
}
+ @Override
+ protected void consumeManage(Caster> source, long costMultiplier, float knowledge) {
+ if (!source.subtractEnergyCost(-accumulatedMass)) {
+ setDead();
+ }
+ }
+
@Override
public boolean isFriendlyTogether(Affine other) {
return accumulatedMass < 4;
@@ -116,21 +111,20 @@ public class DarkVortexSpell extends AttractiveSpell implements ProjectileDelega
public void generateParticles(Caster> source) {
super.generateParticles(source);
- float radius = (float)getEventHorizonRadius();
-
- particlEffect.update(getUuid(), source, spawner -> {
- spawner.addParticle(new SphereParticleEffect(UParticles.SPHERE, 0x000000, 0.99F, radius, SPHERE_OFFSET), source.getOriginVector(), Vec3d.ZERO);
- }).ifPresent(p -> {
- p.setAttribute(Attachment.ATTR_RADIUS, radius);
- p.setAttribute(Attachment.ATTR_OPACITY, 2F);
- });
- particlEffect.update(getUuid(), "_ring", source, spawner -> {
- spawner.addParticle(new SphereParticleEffect(UParticles.DISK, 0xAAAAAA, 0.4F, radius + 1, SPHERE_OFFSET), getOrigin(source), Vec3d.ZERO);
- }).ifPresent(p -> {
- p.setAttribute(Attachment.ATTR_RADIUS, radius * 0F);
- });
-
- source.spawnParticles(ParticleTypes.SMOKE, 3);
+ if (getEventHorizonRadius() > 0.3) {
+ double range = getDrawDropOffRange(source);
+ Vec3d origin = getOrigin(source);
+ source.spawnParticles(origin, new Sphere(false, range), 1, p -> {
+ if (!source.asWorld().isAir(BlockPos.ofFloored(p))) {
+ source.addParticle(
+ new FollowingParticleEffect(UParticles.HEALTH_DRAIN, origin, 0.4F)
+ .withChild(ParticleTypes.CAMPFIRE_SIGNAL_SMOKE),
+ p,
+ Vec3d.ZERO
+ );
+ }
+ });
+ }
}
@Override
@@ -162,7 +156,6 @@ public class DarkVortexSpell extends AttractiveSpell implements ProjectileDelega
applyRadialEffect(source, e, e.getPos().distanceTo(origin), radius);
});
}
- setDirty();
});
}
}
@@ -180,8 +173,8 @@ public class DarkVortexSpell extends AttractiveSpell implements ProjectileDelega
// 2. max force (at dist 0) is taken from accumulated mass
// 3. force reaches 0 at distance of drawDropOffRange
- private double getEventHorizonRadius() {
- return Math.sqrt(Math.max(0.001, getMass() - 12));
+ public double getEventHorizonRadius() {
+ return Math.sqrt(Math.max(0.001, getMass() / 3F));
}
private double getAttractiveForce(Caster> source, Entity target) {
@@ -189,8 +182,7 @@ public class DarkVortexSpell extends AttractiveSpell implements ProjectileDelega
}
private double getMass() {
- float pulse = (float)Math.sin(age * 8) / 1F;
- return 10 + Math.min(15, Math.min(0.5F + pulse, (float)Math.exp(age) / 8F - 90) + accumulatedMass / 10F) + pulse;
+ return Math.min(15, 0.1F + accumulatedMass / 10F);
}
@Override
@@ -202,6 +194,11 @@ public class DarkVortexSpell extends AttractiveSpell implements ProjectileDelega
if (distance <= getEventHorizonRadius() + 0.5) {
target.setVelocity(target.getVelocity().multiply(distance / (2 * radius)));
+ if (distance < 1) {
+ target.setVelocity(target.getVelocity().multiply(distance));
+
+ }
+ Living.updateVelocity(target);
@Nullable
Entity master = source.getMaster();
@@ -221,13 +218,19 @@ public class DarkVortexSpell extends AttractiveSpell implements ProjectileDelega
double massOfTarget = AttractionUtils.getMass(target);
- accumulatedMass += massOfTarget;
- setDirty();
+ if (!source.isClient() && massOfTarget != 0) {
+ accumulatedMass += massOfTarget;
+ setDirty();
+ }
+
target.damage(source.damageOf(UDamageTypes.GAVITY_WELL_RECOIL, source), Integer.MAX_VALUE);
if (!(target instanceof PlayerEntity)) {
target.discard();
source.asWorld().playSound(null, source.getOrigin(), USounds.ENCHANTMENT_CONSUMPTION_CONSUME, SoundCategory.AMBIENT, 2, 0.02F);
}
+ if (target.isAlive()) {
+ target.damage(source.asEntity().getDamageSources().outOfWorld(), Integer.MAX_VALUE);
+ }
source.subtractEnergyCost(-massOfTarget * 10);
source.asWorld().playSound(null, source.getOrigin(), USounds.AMBIENT_DARK_VORTEX_MOOD, SoundCategory.AMBIENT, 2, 0.02F);
@@ -243,14 +246,12 @@ public class DarkVortexSpell extends AttractiveSpell implements ProjectileDelega
@Override
public void toNBT(NbtCompound compound) {
super.toNBT(compound);
- compound.putInt("age", age);
compound.putFloat("accumulatedMass", accumulatedMass);
}
@Override
public void fromNBT(NbtCompound compound) {
super.fromNBT(compound);
- age = compound.getInt("age");
accumulatedMass = compound.getFloat("accumulatedMass");
}
}
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/DisperseIllusionSpell.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/DisperseIllusionSpell.java
index 4806d10d..9affae3e 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/DisperseIllusionSpell.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/DisperseIllusionSpell.java
@@ -12,7 +12,7 @@ import com.minelittlepony.unicopia.util.shape.Sphere;
import net.minecraft.util.math.Vec3d;
/**
- * An area-effect spell that disperses illussions.
+ * An area-effect spell that disperses illusions.
*/
public class DisperseIllusionSpell extends AbstractAreaEffectSpell {
protected DisperseIllusionSpell(CustomisedSpellType> type) {
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/DisplacementSpell.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/DisplacementSpell.java
index 14d3abf8..a58b77de 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/DisplacementSpell.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/DisplacementSpell.java
@@ -6,8 +6,6 @@ import com.minelittlepony.unicopia.ability.magic.spell.*;
import com.minelittlepony.unicopia.ability.magic.spell.trait.Trait;
import com.minelittlepony.unicopia.entity.EntityReference;
import com.minelittlepony.unicopia.entity.damage.UDamageTypes;
-import com.minelittlepony.unicopia.entity.mob.CastSpellEntity;
-import com.minelittlepony.unicopia.particle.ParticleHandle.Attachment;
import com.minelittlepony.unicopia.projectile.MagicProjectileEntity;
import com.minelittlepony.unicopia.projectile.ProjectileDelegate;
@@ -16,7 +14,7 @@ import net.minecraft.nbt.NbtCompound;
import net.minecraft.util.hit.EntityHitResult;
import net.minecraft.util.math.Vec3d;
-public class DisplacementSpell extends AbstractSpell implements HomingSpell, PlaceableSpell.PlacementDelegate, ProjectileDelegate.EntityHitListener {
+public class DisplacementSpell extends AbstractSpell implements HomingSpell, ProjectileDelegate.EntityHitListener {
private final EntityReference target = new EntityReference<>();
@@ -67,19 +65,6 @@ public class DisplacementSpell extends AbstractSpell implements HomingSpell, Pla
originator.subtractEnergyCost(destinationPos.distanceTo(sourcePos) / 20F);
}
- @Override
- public void onPlaced(Caster> source, PlaceableSpell parent, CastSpellEntity entity) {
-
- }
-
- @Override
- public void updatePlacement(Caster> caster, PlaceableSpell parent) {
- parent.getParticleEffectAttachment(caster).ifPresent(attachment -> {
- float r = 3 - (1 - ((ticks + 10) / 20F)) * 3;
- attachment.setAttribute(Attachment.ATTR_RADIUS, r);
- });
- }
-
private void teleport(Caster> source, Entity entity, Vec3d pos, Vec3d vel) {
entity.teleport(pos.x, pos.y, pos.z);
entity.setVelocity(vel);
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/FireBoltSpell.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/FireBoltSpell.java
index c2324a5f..a76ef43b 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/FireBoltSpell.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/FireBoltSpell.java
@@ -48,7 +48,7 @@ public class FireBoltSpell extends AbstractSpell implements HomingSpell,
if (caster instanceof MagicProjectileEntity && getTraits().get(Trait.FOCUS) >= 50) {
caster.findAllEntitiesInRange(
getTraits().get(Trait.FOCUS) - 49,
- EntityPredicates.VALID_LIVING_ENTITY.and(TargetSelecter.notOwnerOrFriend(this, caster))
+ EntityPredicates.VALID_LIVING_ENTITY.and(TargetSelecter.validTarget(this, caster))
).findFirst().ifPresent(target -> {
((MagicProjectileEntity)caster).setHomingTarget(target);
});
@@ -60,7 +60,7 @@ public class FireBoltSpell extends AbstractSpell implements HomingSpell,
if (getTraits().get(Trait.FOCUS) >= 50 && target.getOrEmpty(caster.asWorld()).isEmpty()) {
target.set(caster.findAllEntitiesInRange(
getTraits().get(Trait.FOCUS) - 49,
- EntityPredicates.VALID_LIVING_ENTITY.and(TargetSelecter.notOwnerOrFriend(this, caster))
+ EntityPredicates.VALID_LIVING_ENTITY.and(TargetSelecter.validTarget(this, caster))
).findFirst().orElse(null));
}
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/HydrophobicSpell.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/HydrophobicSpell.java
index 5ff51120..d88d9dd0 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/HydrophobicSpell.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/HydrophobicSpell.java
@@ -5,14 +5,16 @@ import java.util.Set;
import com.minelittlepony.unicopia.USounds;
import com.minelittlepony.unicopia.ability.magic.Caster;
-import com.minelittlepony.unicopia.ability.magic.CasterView;
+import com.minelittlepony.unicopia.ability.magic.spell.CastingMethod;
import com.minelittlepony.unicopia.ability.magic.spell.Situation;
+import com.minelittlepony.unicopia.ability.magic.spell.Spell;
import com.minelittlepony.unicopia.ability.magic.spell.trait.SpellTraits;
import com.minelittlepony.unicopia.ability.magic.spell.trait.Trait;
import com.minelittlepony.unicopia.advancement.UCriteria;
import com.minelittlepony.unicopia.entity.player.Pony;
import com.minelittlepony.unicopia.particle.UParticles;
import com.minelittlepony.unicopia.projectile.MagicProjectileEntity;
+import com.minelittlepony.unicopia.server.world.Ether;
import com.minelittlepony.unicopia.util.NbtSerialisable;
import com.minelittlepony.unicopia.util.shape.*;
@@ -21,8 +23,10 @@ import net.minecraft.fluid.*;
import net.minecraft.nbt.*;
import net.minecraft.state.property.Properties;
import net.minecraft.registry.tag.TagKey;
+import net.minecraft.server.world.ServerWorld;
import net.minecraft.util.math.BlockPos;
import net.minecraft.util.math.Vec3d;
+import net.minecraft.world.BlockView;
import net.minecraft.world.World;
public class HydrophobicSpell extends AbstractSpell {
@@ -41,16 +45,15 @@ public class HydrophobicSpell extends AbstractSpell {
}
@Override
- public boolean apply(Caster> source) {
- if (getTraits().get(Trait.GENEROSITY) > 0) {
- return toPlaceable().apply(source);
+ public Spell prepareForCast(Caster> caster, CastingMethod method) {
+ if ((method == CastingMethod.DIRECT || method == CastingMethod.STAFF) && getTraits().get(Trait.GENEROSITY) > 0) {
+ return toPlaceable();
}
- return super.apply(source);
+ return this;
}
@Override
public boolean tick(Caster> source, Situation situation) {
-
if (!source.isClient()) {
World world = source.asWorld();
@@ -86,7 +89,11 @@ public class HydrophobicSpell extends AbstractSpell {
setDead();
}
- source.spawnParticles(new Sphere(true, getRange(source)), 10, pos -> {
+ double range = getRange(source);
+ var entry = Ether.get(source.asWorld()).getOrCreate(this, source);
+ entry.radius = (float)range;
+
+ source.spawnParticles(new Sphere(true, range), 10, pos -> {
BlockPos bp = BlockPos.ofFloored(pos);
if (source.asWorld().getFluidState(bp.up()).isIn(affectedFluid)) {
source.addParticle(UParticles.RAIN_DROPS, pos, Vec3d.ZERO);
@@ -107,6 +114,7 @@ public class HydrophobicSpell extends AbstractSpell {
@Override
protected void onDestroyed(Caster> caster) {
+ Ether.get(caster.asWorld()).remove(this, caster);
storedFluidPositions.removeIf(entry -> {
entry.restore(caster.asWorld());
return true;
@@ -162,13 +170,20 @@ public class HydrophobicSpell extends AbstractSpell {
}
}
- public boolean blocksFlow(Caster> caster, BlockPos pos, FluidState fluid) {
- return fluid.isIn(affectedFluid) && pos.isWithinDistance(caster.getOrigin(), getRange(caster) + 1);
+ public boolean blocksFlow(Ether.Entry> entry, Vec3d center, BlockPos pos, FluidState fluid) {
+ return fluid.isIn(affectedFluid) && pos.isWithinDistance(center, (double)entry.radius + 1);
}
- public static boolean blocksFluidFlow(CasterView world, BlockPos pos, FluidState state) {
- return world.findAllSpellsInRange(pos, 500, SpellType.HYDROPHOBIC).anyMatch(pair -> {
- return pair.getValue().blocksFlow(pair.getKey(), pos, state);
- });
+ public static boolean blocksFluidFlow(BlockView world, BlockPos pos, FluidState state) {
+ if (world instanceof ServerWorld sw) {
+ return Ether.get(sw).anyMatch(SpellType.HYDROPHOBIC, entry -> {
+ var spell = entry.getSpell();
+ var target = entry.entity.getTarget().orElse(null);
+ return spell != null && target != null && spell.blocksFlow(entry, target.pos(), pos, state);
+ });
+ }
+
+ return false;
+
}
}
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/PortalSpell.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/PortalSpell.java
index 680a0ff1..82a66ad9 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/PortalSpell.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/PortalSpell.java
@@ -1,6 +1,9 @@
package com.minelittlepony.unicopia.ability.magic.spell.effect;
import java.util.Optional;
+import java.util.UUID;
+
+import org.jetbrains.annotations.Nullable;
import com.minelittlepony.unicopia.USounds;
import com.minelittlepony.unicopia.Unicopia;
@@ -9,9 +12,9 @@ import com.minelittlepony.unicopia.ability.magic.spell.*;
import com.minelittlepony.unicopia.ability.magic.spell.trait.SpellTraits;
import com.minelittlepony.unicopia.ability.magic.spell.trait.Trait;
import com.minelittlepony.unicopia.entity.EntityReference;
+import com.minelittlepony.unicopia.entity.Living;
import com.minelittlepony.unicopia.entity.mob.CastSpellEntity;
import com.minelittlepony.unicopia.particle.*;
-import com.minelittlepony.unicopia.particle.ParticleHandle.Attachment;
import com.minelittlepony.unicopia.server.world.Ether;
import com.minelittlepony.unicopia.util.shape.*;
@@ -20,8 +23,10 @@ import net.minecraft.block.Blocks;
import net.minecraft.entity.Entity;
import net.minecraft.entity.LivingEntity;
import net.minecraft.nbt.NbtCompound;
-import net.minecraft.particle.ParticleEffect;
+import net.minecraft.network.packet.s2c.play.PositionFlag;
import net.minecraft.particle.ParticleTypes;
+import net.minecraft.server.world.ServerWorld;
+import net.minecraft.util.math.BlockPos;
import net.minecraft.util.math.MathHelper;
import net.minecraft.util.math.Vec3d;
import net.minecraft.world.WorldEvents;
@@ -34,12 +39,14 @@ public class PortalSpell extends AbstractSpell implements PlaceableSpell.Placeme
.build();
private static final Shape PARTICLE_AREA = new Sphere(true, 2, 1, 1, 0);
+ @Nullable
+ private UUID targetPortalId;
+ private float targetPortalPitch;
+ private float targetPortalYaw;
private final EntityReference teleportationTarget = new EntityReference<>();
private boolean publishedPosition;
- private final ParticleHandle particleEffect = new ParticleHandle();
-
private float pitch;
private float yaw;
@@ -49,6 +56,39 @@ public class PortalSpell extends AbstractSpell implements PlaceableSpell.Placeme
super(type);
}
+ public boolean isLinked() {
+ return teleportationTarget.getTarget().isPresent();
+ }
+
+ public Optional> getTarget() {
+ return teleportationTarget.getTarget();
+ }
+
+ public float getPitch() {
+ return pitch;
+ }
+
+ public float getYaw() {
+ return yaw;
+ }
+
+ public float getTargetPitch() {
+ return targetPortalPitch;
+ }
+
+ public float getTargetYaw() {
+ return targetPortalYaw;
+ }
+
+ public float getYawDifference() {
+ return MathHelper.wrapDegrees(180 + targetPortalYaw - yaw);
+ }
+
+ @SuppressWarnings("unchecked")
+ private Optional> getTarget(Caster> source) {
+ return getTarget().map(target -> Ether.get(source.asWorld()).get((SpellType)getType(), target, targetPortalId));
+ }
+
@Override
public boolean apply(Caster> caster) {
setOrientation(caster.asEntity().getPitch(), caster.asEntity().getYaw());
@@ -60,29 +100,12 @@ public class PortalSpell extends AbstractSpell implements PlaceableSpell.Placeme
if (situation == Situation.GROUND) {
if (source.isClient()) {
- Vec3d origin = source.getOriginVector();
-
- ParticleEffect effect = teleportationTarget.getTarget()
- .map(target -> {
- getType();
- return (ParticleEffect)new FollowingParticleEffect(UParticles.HEALTH_DRAIN, target.pos(), 0.2F).withChild(ParticleTypes.ELECTRIC_SPARK);
- })
- .orElse(ParticleTypes.ELECTRIC_SPARK);
-
- source.spawnParticles(origin, particleArea, 5, pos -> {
- source.addParticle(effect, pos, Vec3d.ZERO);
- });
-
- teleportationTarget.getTarget().ifPresentOrElse(target -> {
- particleEffect.update(getUuid(), source, spawner -> {
- spawner.addParticle(new SphereParticleEffect(UParticles.DISK, getType().getColor(), 0.8F, 1.8F, new Vec3d(-pitch + 90, -yaw, 0)), source.getOriginVector(), Vec3d.ZERO);
- });
- }, () -> {
- particleEffect.destroy();
+ source.spawnParticles(particleArea, 5, pos -> {
+ source.addParticle(ParticleTypes.ELECTRIC_SPARK, pos, Vec3d.ZERO);
});
} else {
- teleportationTarget.getTarget().ifPresent(target -> {
- if (Ether.get(source.asWorld()).getEntry(getType(), target.uuid()).isEmpty()) {
+ getTarget().ifPresent(target -> {
+ if (Ether.get(source.asWorld()).get(getType(), target, targetPortalId) == null) {
Unicopia.LOGGER.debug("Lost sibling, breaking connection to " + target.uuid());
teleportationTarget.set(null);
setDirty();
@@ -96,37 +119,56 @@ public class PortalSpell extends AbstractSpell implements PlaceableSpell.Placeme
);
}
- if (!publishedPosition) {
- publishedPosition = true;
- Ether.Entry entry = Ether.get(source.asWorld()).put(getType(), source);
- entry.pitch = pitch;
- entry.yaw = yaw;
- }
+ var entry = Ether.get(source.asWorld()).getOrCreate(this, source);
+ entry.pitch = pitch;
+ entry.yaw = yaw;
+ Ether.get(source.asWorld()).markDirty();
}
return !isDead();
}
- private void tickWithTargetLink(Caster> source, Ether.Entry destination) {
+ private void tickWithTargetLink(Caster> source, Ether.Entry> destination) {
+
+ if (!MathHelper.approximatelyEquals(targetPortalPitch, destination.pitch)) {
+ targetPortalPitch = destination.pitch;
+ setDirty();
+ }
+ if (!MathHelper.approximatelyEquals(targetPortalYaw, destination.yaw)) {
+ targetPortalYaw = destination.yaw;
+ setDirty();
+ }
destination.entity.getTarget().ifPresent(target -> {
source.findAllEntitiesInRange(1).forEach(entity -> {
- if (!entity.hasPortalCooldown() && entity.timeUntilRegen <= 0) {
+ if (!entity.hasPortalCooldown()) {
+
+ float approachYaw = Math.abs(MathHelper.wrapDegrees(entity.getYaw() - this.yaw));
+ if (approachYaw > 80) {
+ return;
+ }
+
Vec3d offset = entity.getPos().subtract(source.getOriginVector());
- float yawDifference = pitch < 15 ? (180 - yaw + destination.yaw) : 0;
- Vec3d dest = target.pos().add(offset.rotateY(yawDifference * MathHelper.RADIANS_PER_DEGREE)).add(0, 0.05, 0);
+ float yawDifference = pitch < 15 ? getYawDifference() : 0;
+ Vec3d dest = target.pos().add(offset.rotateY(yawDifference * MathHelper.RADIANS_PER_DEGREE)).add(0, 0.1, 0);
+
+ if (entity.getWorld().isTopSolid(BlockPos.ofFloored(dest).up(), entity)) {
+ dest = dest.add(0, 1, 0);
+ }
entity.resetPortalCooldown();
- entity.timeUntilRegen = 100;
- entity.setYaw(entity.getYaw() + yawDifference);
+ float yaw = MathHelper.wrapDegrees(entity.getYaw() + yawDifference);
+
entity.setVelocity(entity.getVelocity().rotateY(yawDifference * MathHelper.RADIANS_PER_DEGREE));
entity.getWorld().playSoundFromEntity(null, entity, USounds.ENTITY_PLAYER_UNICORN_TELEPORT, entity.getSoundCategory(), 1, 1);
- entity.teleport(dest.x, dest.y, dest.z);
+ entity.teleport((ServerWorld)entity.getWorld(), dest.x, dest.y, dest.z, PositionFlag.VALUES, yaw, entity.getPitch());
entity.getWorld().playSoundFromEntity(null, entity, USounds.ENTITY_PLAYER_UNICORN_TELEPORT, entity.getSoundCategory(), 1, 1);
setDirty();
+ Living.updateVelocity(entity);
+
if (!source.subtractEnergyCost(Math.sqrt(entity.getPos().subtract(dest).length()))) {
setDead();
}
@@ -142,20 +184,15 @@ public class PortalSpell extends AbstractSpell implements PlaceableSpell.Placeme
return;
}
- Ether ether = Ether.get(source.asWorld());
- ether.getEntries(getType())
- .stream()
- .filter(entry -> entry.isAvailable() && !entry.entity.referenceEquals(source.asEntity()) && entry.entity.isSet())
- .findAny()
- .ifPresent(entry -> {
+ Ether.get(source.asWorld()).anyMatch(getType(), entry -> {
+ if (entry.isAvailable() && !entry.entity.referenceEquals(source.asEntity()) && entry.entity.isSet()) {
entry.setTaken(true);
teleportationTarget.copyFrom(entry.entity);
+ targetPortalId = entry.getSpellId();
setDirty();
- });
- }
-
- private Optional getTarget(Caster> source) {
- return teleportationTarget.getTarget().flatMap(target -> Ether.get(source.asWorld()).getEntry(getType(), target.uuid()));
+ }
+ return false;
+ });
}
@Override
@@ -174,41 +211,40 @@ public class PortalSpell extends AbstractSpell implements PlaceableSpell.Placeme
LivingEntity caster = source.getMaster();
Vec3d targetPos = caster.getRotationVector().multiply(3).add(caster.getEyePos());
parent.setOrientation(pitch, yaw);
- entity.setPos(targetPos.x, caster.getY() + 1.5, targetPos.z);
- }
-
- @Override
- public void updatePlacement(Caster> source, PlaceableSpell parent) {
- parent.getParticleEffectAttachment(source).ifPresent(attachment -> {
- attachment.setAttribute(Attachment.ATTR_RADIUS, 2);
- attachment.setAttribute(Attachment.ATTR_OPACITY, 0.92F);
- });
+ entity.setPos(targetPos.x, Math.abs(pitch) > 15 ? targetPos.y : caster.getPos().y, targetPos.z);
}
@Override
protected void onDestroyed(Caster> caster) {
- particleEffect.destroy();
Ether ether = Ether.get(caster.asWorld());
- ether.remove(getType(), caster.asEntity().getUuid());
+ ether.remove(getType(), caster);
getTarget(caster).ifPresent(e -> e.setTaken(false));
}
@Override
public void toNBT(NbtCompound compound) {
super.toNBT(compound);
+ if (targetPortalId != null) {
+ compound.putUuid("targetPortalId", targetPortalId);
+ }
compound.putBoolean("publishedPosition", publishedPosition);
compound.put("teleportationTarget", teleportationTarget.toNBT());
compound.putFloat("pitch", pitch);
compound.putFloat("yaw", yaw);
+ compound.putFloat("targetPortalPitch", targetPortalPitch);
+ compound.putFloat("targetPortalYaw", targetPortalYaw);
}
@Override
public void fromNBT(NbtCompound compound) {
super.fromNBT(compound);
+ targetPortalId = compound.containsUuid("targetPortalId") ? compound.getUuid("targetPortalId") : null;
publishedPosition = compound.getBoolean("publishedPosition");
teleportationTarget.fromNBT(compound.getCompound("teleportationTarget"));
pitch = compound.getFloat("pitch");
yaw = compound.getFloat("yaw");
+ targetPortalPitch = compound.getFloat("targetPortalPitch");
+ targetPortalYaw = compound.getFloat("targetPortalYaw");
particleArea = PARTICLE_AREA.rotate(
pitch * MathHelper.RADIANS_PER_DEGREE,
(180 - yaw) * MathHelper.RADIANS_PER_DEGREE
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/ShieldSpell.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/ShieldSpell.java
index 61cc6e79..04831ef6 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/ShieldSpell.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/ShieldSpell.java
@@ -1,5 +1,7 @@
package com.minelittlepony.unicopia.ability.magic.spell.effect;
+import java.util.Optional;
+
import com.minelittlepony.unicopia.Affinity;
import com.minelittlepony.unicopia.USounds;
import com.minelittlepony.unicopia.Unicopia;
@@ -10,11 +12,9 @@ import com.minelittlepony.unicopia.ability.magic.spell.Spell;
import com.minelittlepony.unicopia.ability.magic.spell.trait.SpellTraits;
import com.minelittlepony.unicopia.ability.magic.spell.trait.Trait;
import com.minelittlepony.unicopia.entity.player.Pony;
+import com.minelittlepony.unicopia.particle.LightningBoltParticleEffect;
import com.minelittlepony.unicopia.particle.MagicParticleEffect;
-import com.minelittlepony.unicopia.particle.ParticleHandle;
-import com.minelittlepony.unicopia.particle.SphereParticleEffect;
-import com.minelittlepony.unicopia.particle.UParticles;
-import com.minelittlepony.unicopia.particle.ParticleHandle.Attachment;
+import com.minelittlepony.unicopia.particle.ParticleUtils;
import com.minelittlepony.unicopia.projectile.ProjectileUtil;
import com.minelittlepony.unicopia.util.shape.Sphere;
@@ -30,6 +30,7 @@ import net.minecraft.entity.passive.PassiveEntity;
import net.minecraft.entity.player.PlayerEntity;
import net.minecraft.entity.vehicle.AbstractMinecartEntity;
import net.minecraft.entity.vehicle.BoatEntity;
+import net.minecraft.util.math.MathHelper;
import net.minecraft.util.math.Vec3d;
public class ShieldSpell extends AbstractSpell {
@@ -40,9 +41,16 @@ public class ShieldSpell extends AbstractSpell {
.with(Trait.AIR, 9)
.build();
- protected final ParticleHandle particlEffect = new ParticleHandle();
+ private final TargetSelecter targetSelecter = new TargetSelecter(this).setFilter(this::isValidTarget);
- private final TargetSelecter targetSelecter = new TargetSelecter(this);
+ private float prevRadius;
+ private float radius;
+
+ private float rangeMultiplier;
+ private float targetRangeMultiplier;
+
+ private int prevTicksDying;
+ private int ticksDying;
protected ShieldSpell(CustomisedSpellType> type) {
super(type);
@@ -53,33 +61,27 @@ public class ShieldSpell extends AbstractSpell {
return method == CastingMethod.STAFF || getTraits().get(Trait.GENEROSITY) > 0 ? toPlaceable() : this;
}
- @Override
- protected void onDestroyed(Caster> caster) {
- particlEffect.destroy();
- }
-
@Override
public Affinity getAffinity() {
return getTraits().get(Trait.DARKNESS) > 0 ? Affinity.BAD : Affinity.GOOD;
}
protected void generateParticles(Caster> source) {
- float radius = (float)getDrawDropOffRange(source);
Vec3d origin = getOrigin(source);
source.spawnParticles(origin, new Sphere(true, radius), (int)(radius * 6), pos -> {
source.addParticle(new MagicParticleEffect(getType().getColor()), pos, Vec3d.ZERO);
- });
- particlEffect.update(getUuid(), source, spawner -> {
- spawner.addParticle(new SphereParticleEffect(UParticles.SPHERE, getType().getColor(), 0.3F, radius), origin, Vec3d.ZERO);
- }).ifPresent(p -> {
- p.setAttribute(Attachment.ATTR_RADIUS, radius);
+ if (source.asWorld().random.nextInt(10) == 0 && source.asWorld().random.nextFloat() < source.getCorruption().getScaled(1)) {
+ ParticleUtils.spawnParticle(source.asWorld(), new LightningBoltParticleEffect(true, 3, 2, 0.1F, Optional.empty()), pos, Vec3d.ZERO);
+ }
});
}
@Override
public boolean tick(Caster> source, Situation situation) {
+ prevRadius = radius;
+ radius = (float)getDrawDropOffRange(source);
if (source.isClient()) {
generateParticles(source);
@@ -97,27 +99,53 @@ public class ShieldSpell extends AbstractSpell {
long costMultiplier = applyEntities(source);
if (costMultiplier > 0) {
- double cost = 2 - source.getLevel().getScaled(2);
-
- cost *= costMultiplier / ((1 + source.getLevel().get()) * 3F);
- cost /= knowledge;
- cost += getDrawDropOffRange(source) / 10F;
-
- if (!source.subtractEnergyCost(cost)) {
- setDead();
- }
+ consumeManage(source, costMultiplier, knowledge);
}
return !isDead();
}
+ @Override
+ public void tickDying(Caster> caster) {
+ prevTicksDying = ticksDying;
+ if (ticksDying++ > 25) {
+ super.tickDying(caster);
+ }
+ }
+
+ protected void consumeManage(Caster> source, long costMultiplier, float knowledge) {
+ double cost = 2 - source.getLevel().getScaled(2);
+
+ cost *= costMultiplier / ((1 + source.getLevel().get()) * 3F);
+ cost /= knowledge;
+ cost += radius / 10F;
+
+ if (!source.subtractEnergyCost(cost)) {
+ setDead();
+ }
+ }
+
+ public float getRadius(float tickDelta) {
+ float base = MathHelper.lerp(tickDelta, prevRadius, radius);
+ float scale = MathHelper.clamp(MathHelper.lerp(tickDelta, prevTicksDying, ticksDying), 0, 1);
+ return base * scale;
+ }
+
/**
* Calculates the maximum radius of the shield. aka The area of effect.
*/
public double getDrawDropOffRange(Caster> source) {
- float multiplier = source instanceof Pony pony && pony.asEntity().isSneaking() ? 1 : 2;
+ targetRangeMultiplier = source instanceof Pony pony && pony.asEntity().isSneaking() ? 1 : 2;
+ if (rangeMultiplier < targetRangeMultiplier - 0.1F) {
+ rangeMultiplier += 0.1F;
+ } else if (rangeMultiplier > targetRangeMultiplier + 0.1) {
+ rangeMultiplier -= 0.1F;
+ } else {
+ rangeMultiplier = targetRangeMultiplier;
+ }
+
float min = (source instanceof Pony ? 4 : 6) + getTraits().get(Trait.POWER);
- double range = (min + (source.getLevel().getScaled(source instanceof Pony ? 4 : 40) * (source instanceof Pony ? 2 : 10))) / multiplier;
+ double range = (min + (source.getLevel().getScaled(source instanceof Pony ? 4 : 40) * (source instanceof Pony ? 2 : 10))) / rangeMultiplier;
return range;
}
@@ -146,11 +174,8 @@ public class ShieldSpell extends AbstractSpell {
}
protected long applyEntities(Caster> source) {
- double radius = getDrawDropOffRange(source);
-
Vec3d origin = getOrigin(source);
-
- targetSelecter.getEntities(source, radius, this::isValidTarget).forEach(i -> {
+ targetSelecter.getEntities(source, radius).forEach(i -> {
try {
applyRadialEffect(source, i, i.getPos().distanceTo(origin), radius);
} catch (Throwable e) {
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/TargetSelecter.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/TargetSelecter.java
index b2482157..87119eab 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/TargetSelecter.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/effect/TargetSelecter.java
@@ -10,28 +10,32 @@ import java.util.stream.Stream;
import com.minelittlepony.unicopia.EquinePredicates;
import com.minelittlepony.unicopia.ability.magic.Affine;
import com.minelittlepony.unicopia.ability.magic.Caster;
-import com.minelittlepony.unicopia.ability.magic.SpellPredicate;
import com.minelittlepony.unicopia.ability.magic.spell.Spell;
-import com.minelittlepony.unicopia.entity.player.Pony;
-import com.minelittlepony.unicopia.item.FriendshipBraceletItem;
import net.minecraft.entity.Entity;
+import net.minecraft.predicate.entity.EntityPredicates;
public class TargetSelecter {
-
private final Map targets = new TreeMap<>();
private final Spell spell;
+ private BiPredicate, Entity> filter = (a, b) -> true;
+
public TargetSelecter(Spell spell) {
this.spell = spell;
}
- public Stream getEntities(Caster> source, double radius, BiPredicate, Entity> filter) {
+ public TargetSelecter setFilter(BiPredicate, Entity> filter) {
+ this.filter = filter;
+ return this;
+ }
+
+ public Stream getEntities(Caster> source, double radius) {
targets.values().removeIf(Target::tick);
return source.findAllEntitiesInRange(radius)
- .filter(entity -> entity.isAlive() && !entity.isRemoved() && notOwnerOrFriend(spell, source, entity) && !SpellPredicate.IS_SHIELD_LIKE.isOn(entity))
+ .filter(EntityPredicates.VALID_ENTITY)
.filter(EquinePredicates.EXCEPT_MAGIC_IMMUNE)
- .filter(e -> filter.test(source, e))
+ .filter(entity -> entity != source.asEntity() && validTarget(spell, source, entity) && filter.test(source, entity))
.map(i -> {
targets.computeIfAbsent(i.getUuid(), Target::new);
return i;
@@ -42,30 +46,19 @@ public class TargetSelecter {
return targets.values().stream().filter(Target::canHurt).count();
}
- public static Predicate notOwnerOrFriend(Affine affine, Caster> source) {
- return target -> notOwnerOrFriend(affine, source, target);
+ public static Predicate validTarget(Affine affine, Caster> source) {
+ return target -> validTarget(affine, source, target);
}
- public static Predicate isOwnerOrFriend(Affine affine, Caster> source) {
- return target -> isOwnerOrFriend(affine, source, target);
- }
-
- public static boolean notOwnerOrFriend(Affine affine, Caster> source, Entity target) {
+ public static boolean validTarget(Affine affine, Caster> source, Entity target) {
return !isOwnerOrFriend(affine, source, target);
}
- public static boolean isOwnerOrFriend(Affine affine, Caster> source, Entity target) {
- Entity owner = source.getMaster();
-
- if (affine.isEnemy(source)) {
- return FriendshipBraceletItem.isComrade(source, target);
- }
-
- return FriendshipBraceletItem.isComrade(source, target)
- || (owner != null && (Pony.equal(target, owner) || owner.isConnectedThroughVehicle(target)));
+ public static boolean isOwnerOrFriend(Affine affine, Caster> source, Entity target) {
+ return affine.applyInversion(source, source.isOwnerOrFriend(target));
}
- static final class Target {
+ private static final class Target {
private int cooldown = 20;
Target(UUID id) { }
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/trait/ItemWithTraits.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/trait/ItemWithTraits.java
new file mode 100644
index 00000000..5b6cd107
--- /dev/null
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/trait/ItemWithTraits.java
@@ -0,0 +1,5 @@
+package com.minelittlepony.unicopia.ability.magic.spell.trait;
+
+public interface ItemWithTraits {
+ SpellTraits getDefaultTraits();
+}
diff --git a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/trait/SpellTraits.java b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/trait/SpellTraits.java
index 58c24bc4..17cd9568 100644
--- a/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/trait/SpellTraits.java
+++ b/src/main/java/com/minelittlepony/unicopia/ability/magic/spell/trait/SpellTraits.java
@@ -29,6 +29,7 @@ import net.fabricmc.api.Environment;
import net.minecraft.block.Block;
import net.minecraft.inventory.Inventory;
import net.minecraft.item.Item;
+import net.minecraft.item.SpawnEggItem;
import net.minecraft.item.ItemStack;
import net.minecraft.nbt.NbtCompound;
import net.minecraft.nbt.NbtElement;
@@ -41,6 +42,7 @@ import net.minecraft.registry.Registries;
public final class SpellTraits implements Iterable> {
public static final SpellTraits EMPTY = new SpellTraits(Map.of());
+ private static final SpellTraits SPAWN_EGG_TRAITS = new SpellTraits(Map.of(Trait.LIFE, 20F));
private static Map REGISTRY = new HashMap<>();
static final Map> ITEMS = new HashMap<>();
@@ -220,6 +222,12 @@ public final class SpellTraits implements Iterable> {
}
public static SpellTraits of(Item item) {
+ if (item instanceof ItemWithTraits i) {
+ return i.getDefaultTraits();
+ }
+ if (item instanceof SpawnEggItem) {
+ return SPAWN_EGG_TRAITS;
+ }
return REGISTRY.getOrDefault(Registries.ITEM.getId(item), EMPTY);
}
diff --git a/src/main/java/com/minelittlepony/unicopia/block/BaseZapAppleLeavesBlock.java b/src/main/java/com/minelittlepony/unicopia/block/BaseZapAppleLeavesBlock.java
index 0b56490b..902b5a11 100644
--- a/src/main/java/com/minelittlepony/unicopia/block/BaseZapAppleLeavesBlock.java
+++ b/src/main/java/com/minelittlepony/unicopia/block/BaseZapAppleLeavesBlock.java
@@ -13,7 +13,7 @@ import net.minecraft.util.math.*;
import net.minecraft.util.math.random.Random;
import net.minecraft.world.*;
-public class BaseZapAppleLeavesBlock extends LeavesBlock implements TintedBlock {
+public class BaseZapAppleLeavesBlock extends LeavesBlock implements TintedBlock, ZapStagedBlock {
BaseZapAppleLeavesBlock() {
super(Settings.create()
@@ -29,61 +29,29 @@ public class BaseZapAppleLeavesBlock extends LeavesBlock implements TintedBlock
}
@Override
- public boolean hasRandomTicks(BlockState state) {
- return !state.get(PERSISTENT);
- }
-
- @Override
- public void randomTick(BlockState state, ServerWorld world, BlockPos pos, Random random) {
- super.randomTick(state, world, pos, random);
- tryAdvanceStage(state, world, pos, random);
- }
-
- @Override
- public BlockState getStateForNeighborUpdate(BlockState state, Direction direction, BlockState neighborState, WorldAccess world, BlockPos pos, BlockPos neighborPos) {
- if (state.get(PERSISTENT)) {
- return state;
+ public void onBlockAdded(BlockState state, World world, BlockPos pos, BlockState oldState, boolean notify) {
+ if (state.get(PERSISTENT) || oldState.isOf(state.getBlock())) {
+ return;
}
-
- if (world instanceof ServerWorld sw) {
- ZapAppleStageStore store = ZapAppleStageStore.get(sw);
- ZapAppleStageStore.Stage currentStage = store.getStage();
- if (currentStage == ZapAppleStageStore.Stage.HIBERNATING) {
- return currentStage.getNewState(state);
- }
- }
-
- return state;
+ updateStage(state, world, pos);
}
@Override
public void scheduledTick(BlockState state, ServerWorld world, BlockPos pos, Random random) {
super.scheduledTick(state, world, pos, random);
- tryAdvanceStage(state, world, pos, random);
- if (!state.get(PERSISTENT)) {
- world.scheduleBlockTick(pos, this, 1);
- }
- }
-
- private void tryAdvanceStage(BlockState state, ServerWorld world, BlockPos pos, Random random) {
if (state.get(PERSISTENT)) {
return;
}
-
- ZapAppleStageStore store = ZapAppleStageStore.get(world);
- ZapAppleStageStore.Stage newStage = store.getStage();
- if (!world.isDay() && getStage(state).mustChangeIntoInstantly(newStage)) {
- world.setBlockState(pos, newStage.getNewState(state));
- onStageChanged(store, newStage, world, state, pos, random);
- }
+ tryAdvanceStage(state, world, pos, random);
}
- protected ZapAppleStageStore.Stage getStage(BlockState state) {
+ @Override
+ public ZapAppleStageStore.Stage getStage(BlockState state) {
return ZapAppleStageStore.Stage.FLOWERING;
}
@Override
- protected boolean shouldDecay(BlockState state) {
+ protected final boolean shouldDecay(BlockState state) {
return false;
}
@@ -114,40 +82,10 @@ public class BaseZapAppleLeavesBlock extends LeavesBlock implements TintedBlock
@Override
public int getTint(BlockState state, @Nullable BlockRenderView world, @Nullable BlockPos pos, int foliageColor) {
-
if (pos == null) {
return 0x4C7EFA;
}
return TintedBlock.blend(TintedBlock.rotate(foliageColor, 2), 0x0000FF, 0.3F);
}
-
- static void onStageChanged(ZapAppleStageStore store, ZapAppleStageStore.Stage stage, ServerWorld world, BlockState state, BlockPos pos, Random random) {
- boolean mustFruit = Random.create(state.getRenderingSeed(pos)).nextInt(5) < 2;
- BlockState below = world.getBlockState(pos.down());
-
- if (world.isAir(pos.down())) {
- if (stage == ZapAppleStageStore.Stage.FRUITING && mustFruit) {
- world.setBlockState(pos.down(), UBlocks.ZAP_BULB.getDefaultState(), Block.NOTIFY_ALL);
- store.triggerLightningStrike(pos);
- }
- }
-
- if (stage != ZapAppleStageStore.Stage.HIBERNATING && world.getRandom().nextInt(10) == 0) {
- store.triggerLightningStrike(pos);
- }
-
- if (stage == ZapAppleStageStore.Stage.RIPE) {
- if (below.isOf(UBlocks.ZAP_BULB)) {
- world.setBlockState(pos.down(), UBlocks.ZAP_APPLE.getDefaultState(), Block.NOTIFY_ALL);
- store.playMoonEffect(pos);
- }
- }
-
- if (mustFruit && stage == ZapAppleStageStore.Stage.HIBERNATING) {
- if (below.isOf(UBlocks.ZAP_APPLE) || below.isOf(UBlocks.ZAP_BULB)) {
- world.setBlockState(pos.down(), Blocks.AIR.getDefaultState());
- }
- }
- }
}
diff --git a/src/main/java/com/minelittlepony/unicopia/block/CuringJokeBlock.java b/src/main/java/com/minelittlepony/unicopia/block/CuringJokeBlock.java
new file mode 100644
index 00000000..61658a29
--- /dev/null
+++ b/src/main/java/com/minelittlepony/unicopia/block/CuringJokeBlock.java
@@ -0,0 +1,47 @@
+package com.minelittlepony.unicopia.block;
+
+import com.minelittlepony.unicopia.ability.EarthPonyGrowAbility.Growable;
+import com.minelittlepony.unicopia.entity.mob.IgnominiousBulbEntity;
+import com.minelittlepony.unicopia.particle.MagicParticleEffect;
+
+import net.minecraft.block.BlockState;
+import net.minecraft.block.FlowerBlock;
+import net.minecraft.client.util.ParticleUtil;
+import net.minecraft.entity.Dismounting;
+import net.minecraft.entity.effect.StatusEffect;
+import net.minecraft.util.math.BlockPos;
+import net.minecraft.util.math.random.Random;
+import net.minecraft.world.World;
+
+public class CuringJokeBlock extends FlowerBlock implements Growable {
+ public CuringJokeBlock(StatusEffect suspiciousStewEffect, int effectDuration, Settings settings) {
+ super(suspiciousStewEffect, effectDuration, settings);
+ }
+
+ @Override
+ public void randomDisplayTick(BlockState state, World world, BlockPos pos, Random random) {
+ for (int i = 0; i < 3; i++) {
+ ParticleUtil.spawnParticle(world, pos, random, new MagicParticleEffect(0x3388EE));
+ }
+ }
+
+ @Override
+ public boolean grow(World world, BlockState state, BlockPos pos) {
+ var otherFlowers = BlockPos.streamOutwards(pos, 16, 16, 16)
+ .filter(p -> world.getBlockState(p).isOf(this))
+ .map(BlockPos::toImmutable)
+ .toList();
+
+ IgnominiousBulbEntity bulb = new IgnominiousBulbEntity(world);
+ bulb.setBaby(true);
+ bulb.updatePositionAndAngles(pos.getX() + 0.5, pos.getY(), pos.getZ() + 0.5, 0, 0);
+
+ if (Dismounting.canPlaceEntityAt(world, bulb, bulb.getBoundingBox())) {
+ otherFlowers.forEach(p -> world.breakBlock(p, false));
+ world.spawnEntity(bulb);
+ return true;
+ }
+
+ return false;
+ }
+}
diff --git a/src/main/java/com/minelittlepony/unicopia/block/EdibleBlock.java b/src/main/java/com/minelittlepony/unicopia/block/EdibleBlock.java
new file mode 100644
index 00000000..b450c828
--- /dev/null
+++ b/src/main/java/com/minelittlepony/unicopia/block/EdibleBlock.java
@@ -0,0 +1,229 @@
+package com.minelittlepony.unicopia.block;
+
+import java.util.ArrayList;
+import java.util.List;
+import java.util.function.Function;
+
+import org.jetbrains.annotations.Nullable;
+
+import com.minelittlepony.unicopia.USounds;
+import com.minelittlepony.unicopia.block.state.StateUtil;
+import com.minelittlepony.unicopia.entity.player.Pony;
+
+import net.fabricmc.fabric.api.event.player.UseBlockCallback;
+import net.fabricmc.fabric.api.registry.FlammableBlockRegistry;
+import net.minecraft.block.Block;
+import net.minecraft.block.BlockState;
+import net.minecraft.block.Blocks;
+import net.minecraft.block.HayBlock;
+import net.minecraft.block.ShapeContext;
+import net.minecraft.entity.player.PlayerEntity;
+import net.minecraft.item.ItemStack;
+import net.minecraft.registry.Registries;
+import net.minecraft.registry.tag.ItemTags;
+import net.minecraft.sound.SoundCategory;
+import net.minecraft.state.StateManager;
+import net.minecraft.state.property.BooleanProperty;
+import net.minecraft.util.ActionResult;
+import net.minecraft.util.Hand;
+import net.minecraft.util.Identifier;
+import net.minecraft.util.Util;
+import net.minecraft.util.hit.BlockHitResult;
+import net.minecraft.util.math.BlockPos;
+import net.minecraft.util.math.Vec3d;
+import net.minecraft.util.shape.VoxelShape;
+import net.minecraft.util.shape.VoxelShapes;
+import net.minecraft.world.BlockView;
+import net.minecraft.world.World;
+
+public class EdibleBlock extends HayBlock {
+ private static final List REGISTRY = new ArrayList<>();
+
+ static final BooleanProperty TOP_NORTH_EAST = BooleanProperty.of("top_north_east");
+ static final BooleanProperty TOP_NORTH_WEST = BooleanProperty.of("top_north_west");
+ static final BooleanProperty TOP_SOUTH_EAST = BooleanProperty.of("top_south_east");
+ static final BooleanProperty TOP_SOUTH_WEST = BooleanProperty.of("top_south_west");
+
+ static final BooleanProperty BOTTOM_NORTH_EAST = BooleanProperty.of("bottom_north_east");
+ static final BooleanProperty BOTTOM_NORTH_WEST = BooleanProperty.of("bottom_north_west");
+ static final BooleanProperty BOTTOM_SOUTH_EAST = BooleanProperty.of("bottom_south_east");
+ static final BooleanProperty BOTTOM_SOUTH_WEST = BooleanProperty.of("bottom_south_west");
+
+ // [up/down][north/south][west/east]
+ private static final BooleanProperty[] SEGMENTS = {
+ BOTTOM_NORTH_WEST,
+ BOTTOM_NORTH_EAST,
+ BOTTOM_SOUTH_WEST,
+ BOTTOM_SOUTH_EAST,
+ TOP_NORTH_WEST,
+ TOP_NORTH_EAST,
+ TOP_SOUTH_WEST,
+ TOP_SOUTH_EAST
+ };
+ private static final VoxelShape[] SHAPES = {
+ Block.createCuboidShape(0, 0, 0, 8, 8, 8),
+ Block.createCuboidShape(8, 0, 0, 16, 8, 8),
+ Block.createCuboidShape(0, 0, 8, 8, 8, 16),
+ Block.createCuboidShape(8, 0, 8, 16, 8, 16),
+ Block.createCuboidShape(0, 8, 0, 8, 16, 8),
+ Block.createCuboidShape(8, 8, 0, 16, 16, 8),
+ Block.createCuboidShape(0, 8, 8, 8, 16, 16),
+ Block.createCuboidShape(8, 8, 8, 16, 16, 16)
+ };
+ private static final Function SHAPE_CACHE = Util.memoize(state -> {
+ @Nullable
+ VoxelShape shape = null;
+ for (int i = 0; i < SEGMENTS.length; i++) {
+ if (state.get(SEGMENTS[i])) {
+ shape = shape == null ? SHAPES[i] : VoxelShapes.union(shape, SHAPES[i]);
+ }
+ }
+ return shape == null ? VoxelShapes.fullCube() : shape.simplify();
+ });
+
+ static void bootstrap() {
+ UseBlockCallback.EVENT.register((PlayerEntity player, World world, Hand hand, BlockHitResult hitResult) -> {
+ if (!Pony.of(player).getSpecies().isEquine()
+ || (player.shouldCancelInteraction() && (!player.getMainHandStack().isEmpty() || !player.getOffHandStack().isEmpty()))) {
+ return ActionResult.PASS;
+ }
+
+ BlockPos pos = hitResult.getBlockPos();
+ BlockState state = world.getBlockState(pos);
+
+ for (EdibleBlock edibleBlock : REGISTRY) {
+ Block match = edibleBlock.getBaseBlock();
+ if (match != Blocks.AIR && state.isOf(match)) {
+ ActionResult result = StateUtil.copyState(state, edibleBlock.getDefaultState()).onUse(world, player, hand, hitResult);
+
+ if (result.isAccepted()) {
+ return result;
+ }
+ }
+ }
+
+ return ActionResult.PASS;
+ });
+ }
+
+ private final Identifier baseBlock;
+ private final Identifier material;
+
+ public EdibleBlock(Identifier baseBlock, Identifier material, boolean register) {
+ super(Settings.copy(Blocks.HAY_BLOCK));
+ for (BooleanProperty segment : SEGMENTS) {
+ setDefaultState(getDefaultState().with(segment, true));
+ }
+ this.baseBlock = baseBlock;
+ this.material = material;
+ if (register) {
+ REGISTRY.add(this);
+ FlammableBlockRegistry.getDefaultInstance().add(this, 60, 20);
+ }
+ }
+
+ public Block getBaseBlock() {
+ return Registries.BLOCK.get(baseBlock);
+ }
+
+ @Override
+ public String getTranslationKey() {
+ return getBaseBlock().getTranslationKey();
+ }
+
+ @Override
+ protected void appendProperties(StateManager.Builder builder) {
+ super.appendProperties(builder);
+ builder.add(SEGMENTS);
+ }
+
+ @Override
+ public VoxelShape getOutlineShape(BlockState state, BlockView world, BlockPos pos, ShapeContext context) {
+ return SHAPE_CACHE.apply(state);
+ }
+
+ @Override
+ @Deprecated
+ public ActionResult onUse(BlockState state, World world, BlockPos pos, PlayerEntity player, Hand hand, BlockHitResult hit) {
+ if (player.isSpectator()) {
+ return ActionResult.FAIL;
+ }
+
+ ItemStack stack = player.getStackInHand(hand);
+
+ if (!stack.isEmpty() && stack.isOf(Registries.ITEM.get(material))) {
+ BooleanProperty segment = getHitCorner(hit, 1);
+
+ if (!state.get(segment)) {
+ if (!player.isCreative()) {
+ stack.decrement(1);
+ }
+ if (!world.isClient) {
+ state = state.with(segment, true);
+ if (SHAPE_CACHE.apply(state) == VoxelShapes.fullCube()) {
+ state = StateUtil.copyState(state, getBaseBlock().getDefaultState());
+ }
+ world.setBlockState(pos, state);
+ }
+ world.playSound(player, pos, getSoundGroup(state).getPlaceSound(), SoundCategory.BLOCKS);
+
+ return ActionResult.SUCCESS;
+ }
+
+ return ActionResult.FAIL;
+ }
+
+ BooleanProperty corner = getHitCorner(hit, -1);
+
+ if (!state.get(corner)) {
+ return ActionResult.PASS;
+ }
+
+ boolean usingHoe = stack.isIn(ItemTags.HOES);
+
+ if (!usingHoe) {
+ if (!(player.isCreative() || player.getHungerManager().isNotFull()) || !player.isSneaking()) {
+ return ActionResult.FAIL;
+ }
+ }
+
+ if (!world.isClient) {
+ state = state.with(corner, false);
+ if (SHAPE_CACHE.apply(state) == VoxelShapes.fullCube()) {
+ world.removeBlock(pos, false);
+ } else {
+ world.setBlockState(pos, state);
+ }
+ }
+
+ if (usingHoe) {
+ stack.damage(1, player, p -> p.sendToolBreakStatus(hand));
+ dropStack(world, pos, Registries.ITEM.get(material).getDefaultStack());
+ player.playSound(USounds.Vanilla.ITEM_HOE_TILL, 1, 1);
+ } else {
+ player.playSound(USounds.Vanilla.ENTITY_GENERIC_EAT, 1, 1);
+ if (world.random.nextInt(10) == 0) {
+ player.playSound(USounds.Vanilla.ENTITY_PLAYER_BURP, 1, player.getSoundPitch());
+ }
+ player.getHungerManager().add(4, 2.3F);
+ }
+ return ActionResult.SUCCESS;
+ }
+
+ static BooleanProperty getHitCorner(BlockHitResult hit, int direction) {
+ Vec3d pos = hit.getPos().add(Vec3d.of(hit.getSide().getVector()).multiply(direction * 0.001F));
+
+ BlockPos bPos = hit.getBlockPos();
+
+ return SEGMENTS[
+ (4 * getIndex(pos.y, bPos.getY()))
+ + (2 * getIndex(pos.z, bPos.getZ()))
+ + (getIndex(pos.x, bPos.getX()))
+ ];
+ }
+
+ static int getIndex(double axisHit, int tile) {
+ axisHit -= tile;
+ return Math.abs(axisHit) > 0.5 ? 1 : 0;
+ }
+}
diff --git a/src/main/java/com/minelittlepony/unicopia/block/EnchantedFruitBlock.java b/src/main/java/com/minelittlepony/unicopia/block/EnchantedFruitBlock.java
new file mode 100644
index 00000000..c6af9166
--- /dev/null
+++ b/src/main/java/com/minelittlepony/unicopia/block/EnchantedFruitBlock.java
@@ -0,0 +1,24 @@
+package com.minelittlepony.unicopia.block;
+
+import net.minecraft.block.Block;
+import net.minecraft.block.BlockState;
+import net.minecraft.state.StateManager;
+import net.minecraft.state.property.BooleanProperty;
+import net.minecraft.util.math.Direction;
+import net.minecraft.util.shape.VoxelShape;
+
+public class EnchantedFruitBlock extends FruitBlock {
+ static final BooleanProperty ENCHANTED = BooleanProperty.of("enchanted");
+
+ public EnchantedFruitBlock(Settings settings, Direction attachmentFace, Block stem, VoxelShape shape) {
+ super(settings, attachmentFace, stem, shape);
+ setDefaultState(getDefaultState().with(ENCHANTED, false));
+ }
+
+ @Override
+ protected void appendProperties(StateManager.Builder builder) {
+ super.appendProperties(builder);
+ builder.add(ENCHANTED);
+ }
+
+}
diff --git a/src/main/java/com/minelittlepony/unicopia/block/FruitBearingBlock.java b/src/main/java/com/minelittlepony/unicopia/block/FruitBearingBlock.java
index ee2eb14d..b9741cef 100644
--- a/src/main/java/com/minelittlepony/unicopia/block/FruitBearingBlock.java
+++ b/src/main/java/com/minelittlepony/unicopia/block/FruitBearingBlock.java
@@ -62,6 +62,14 @@ public class FruitBearingBlock extends LeavesBlock implements TintedBlock, Bucka
return true;
}
+ protected boolean shouldAdvance(Random random) {
+ return true;
+ }
+
+ protected BlockState getPlacedFruitState(Random random) {
+ return fruit.get().getDefaultState();
+ }
+
@Override
public void randomTick(BlockState state, ServerWorld world, BlockPos pos, Random random) {
super.randomTick(state, world, pos, random);
@@ -70,10 +78,14 @@ public class FruitBearingBlock extends LeavesBlock implements TintedBlock, Bucka
return;
}
- if (world.isDay()) {
+ if (world.getBaseLightLevel(pos, 0) > 8) {
BlockSoundGroup group = getSoundGroup(state);
int steps = FertilizableUtil.getGrowthSteps(world, pos, state, random);
while (steps-- > 0) {
+ if (!shouldAdvance(random)) {
+ continue;
+ }
+
if (state.get(STAGE) == Stage.FRUITING) {
state = state.cycle(AGE);
if (state.get(AGE) > 20) {
@@ -89,7 +101,7 @@ public class FruitBearingBlock extends LeavesBlock implements TintedBlock, Bucka
if (stage == Stage.FRUITING && isPositionValidForFruit(state, pos)) {
if (world.isAir(fruitPosition)) {
- world.setBlockState(fruitPosition, fruit.get().getDefaultState(), Block.NOTIFY_ALL);
+ world.setBlockState(fruitPosition, getPlacedFruitState(random), Block.NOTIFY_ALL);
}
}
diff --git a/src/main/java/com/minelittlepony/unicopia/block/GoldenOakLeavesBlock.java b/src/main/java/com/minelittlepony/unicopia/block/GoldenOakLeavesBlock.java
new file mode 100644
index 00000000..1ae9ed61
--- /dev/null
+++ b/src/main/java/com/minelittlepony/unicopia/block/GoldenOakLeavesBlock.java
@@ -0,0 +1,26 @@
+package com.minelittlepony.unicopia.block;
+
+import java.util.function.Supplier;
+
+import net.minecraft.block.Block;
+import net.minecraft.block.BlockState;
+import net.minecraft.item.ItemStack;
+import net.minecraft.util.math.random.Random;
+
+public class GoldenOakLeavesBlock extends FruitBearingBlock {
+
+ public GoldenOakLeavesBlock(Settings settings, int overlay, Supplier fruit,
+ Supplier rottenFruitSupplier) {
+ super(settings, overlay, fruit, rottenFruitSupplier);
+ }
+
+ @Override
+ protected boolean shouldAdvance(Random random) {
+ return random.nextInt(1000) == 0;
+ }
+
+ @Override
+ protected BlockState getPlacedFruitState(Random random) {
+ return super.getPlacedFruitState(random).with(EnchantedFruitBlock.ENCHANTED, random.nextInt(1000) == 0);
+ }
+}
diff --git a/src/main/java/com/minelittlepony/unicopia/block/UBlocks.java b/src/main/java/com/minelittlepony/unicopia/block/UBlocks.java
index b74d0566..02ddcdb6 100644
--- a/src/main/java/com/minelittlepony/unicopia/block/UBlocks.java
+++ b/src/main/java/com/minelittlepony/unicopia/block/UBlocks.java
@@ -21,6 +21,7 @@ import com.minelittlepony.unicopia.block.cloud.SoggyCloudBlock;
import com.minelittlepony.unicopia.block.cloud.SoggyCloudSlabBlock;
import com.minelittlepony.unicopia.block.cloud.SoggyCloudStairsBlock;
import com.minelittlepony.unicopia.block.cloud.UnstableCloudBlock;
+import com.minelittlepony.unicopia.entity.effect.UEffects;
import com.minelittlepony.unicopia.item.UItems;
import com.minelittlepony.unicopia.item.cloud.CloudBlockItem;
import com.minelittlepony.unicopia.item.group.ItemGroupRegistry;
@@ -28,6 +29,7 @@ import com.minelittlepony.unicopia.server.world.UTreeGen;
import net.fabricmc.fabric.api.object.builder.v1.block.FabricBlockSettings;
import net.fabricmc.fabric.api.registry.FlammableBlockRegistry;
import net.fabricmc.fabric.api.registry.StrippableBlockRegistry;
+import net.fabricmc.loader.api.FabricLoader;
import net.minecraft.block.*;
import net.minecraft.block.AbstractBlock.Settings;
import net.minecraft.block.entity.BlockEntityType;
@@ -78,7 +80,7 @@ public interface UBlocks {
Block PALM_STAIRS = register("palm_stairs", new StairsBlock(PALM_PLANKS.getDefaultState(), Settings.copy(PALM_PLANKS).pistonBehavior(PistonBehavior.NORMAL)), ItemGroups.BUILDING_BLOCKS);
Block PALM_SLAB = register("palm_slab", new SlabBlock(Settings.create().mapColor(PALM_PLANKS.getDefaultMapColor()).strength(2, 3).sounds(BlockSoundGroup.WOOD).pistonBehavior(PistonBehavior.NORMAL)), ItemGroups.BUILDING_BLOCKS);
Block PALM_FENCE = register("palm_fence", new FenceBlock(Settings.create().mapColor(PALM_PLANKS.getDefaultMapColor()).strength(2, 3).sounds(BlockSoundGroup.WOOD).pistonBehavior(PistonBehavior.NORMAL)), ItemGroups.BUILDING_BLOCKS);
- Block PALM_FENCE_GATE = register("palm_fence_gate", new FenceGateBlock(Settings.create().mapColor(PALM_PLANKS.getDefaultMapColor()).strength(2, 3).sounds(BlockSoundGroup.WOOD).pistonBehavior(PistonBehavior.NORMAL), WoodType.OAK), ItemGroups.BUILDING_BLOCKS);
+ Block PALM_FENCE_GATE = register("palm_fence_gate", new FenceGateBlock(Settings.create().mapColor(PALM_PLANKS.getDefaultMapColor()).strength(2, 3).sounds(BlockSoundGroup.WOOD).pistonBehavior(PistonBehavior.NORMAL), UWoodTypes.PALM), ItemGroups.BUILDING_BLOCKS);
Block PALM_DOOR = register("palm_door", new DoorBlock(Settings.create().mapColor(PALM_PLANKS.getDefaultMapColor()).instrument(Instrument.BASS).strength(3.0f).nonOpaque().burnable().pistonBehavior(PistonBehavior.DESTROY), UWoodTypes.PALM.setType()), ItemGroups.FUNCTIONAL);
Block PALM_TRAPDOOR = register("palm_trapdoor", new TrapdoorBlock(Settings.create().mapColor(PALM_PLANKS.getDefaultMapColor()).instrument(Instrument.BASS).strength(3).nonOpaque().allowsSpawning(BlockConstructionUtils::never).burnable(), UWoodTypes.PALM.setType()), ItemGroups.FUNCTIONAL);
Block PALM_PRESSURE_PLATE = register("palm_pressure_plate", new PressurePlateBlock(PressurePlateBlock.ActivationRule.EVERYTHING, Settings.create().mapColor(PALM_PLANKS.getDefaultMapColor()).noCollision().strength(0.5f).sounds(BlockSoundGroup.WOOD).pistonBehavior(PistonBehavior.DESTROY), BlockSetType.OAK), ItemGroups.BUILDING_BLOCKS);
@@ -126,6 +128,16 @@ public interface UBlocks {
Block SOUR_APPLE = register("sour_apple", new FruitBlock(Settings.create().mapColor(MapColor.GREEN), Direction.DOWN, SOUR_APPLE_LEAVES, FruitBlock.DEFAULT_SHAPE));
Block SOUR_APPLE_SPROUT = register("sour_apple_sprout", new SproutBlock(0xE5FFCC88, () -> UItems.SOUR_APPLE_SEEDS, () -> UTreeGen.SOUR_APPLE_TREE.sapling().map(Block::getDefaultState).get()));
+ Block GOLDEN_OAK_LEAVES = register("golden_oak_leaves", new GoldenOakLeavesBlock(FabricBlockSettings.copy(Blocks.OAK_LEAVES),
+ MapColor.GOLD.color,
+ () -> UBlocks.GOLDEN_APPLE,
+ () -> Items.GOLDEN_APPLE.getDefaultStack()
+ ), ItemGroups.NATURAL);
+ Block GOLDEN_APPLE = register("golden_apple", new EnchantedFruitBlock(Settings.create().mapColor(MapColor.GOLD), Direction.DOWN, GOLDEN_OAK_LEAVES, FruitBlock.DEFAULT_SHAPE));
+ Block GOLDEN_OAK_SPROUT = register("golden_oak_sprout", new SproutBlock(0xE5FFCC88, () -> UItems.GOLDEN_OAK_SEEDS, () -> UTreeGen.GOLDEN_APPLE_TREE.sapling().map(Block::getDefaultState).get()));
+
+ Block GOLDEN_OAK_LOG = register("golden_oak_log", BlockConstructionUtils.createLogBlock(MapColor.OFF_WHITE, MapColor.GOLD), ItemGroups.BUILDING_BLOCKS);
+
Block APPLE_PIE = register("apple_pie", new PieBlock(Settings.create().solid().mapColor(MapColor.ORANGE).strength(0.5F).sounds(BlockSoundGroup.WOOL).pistonBehavior(PistonBehavior.DESTROY),
() -> UItems.APPLE_PIE_SLICE,
() -> UItems.APPLE_PIE,
@@ -138,6 +150,13 @@ public interface UBlocks {
Block PLUNDER_VINE = register("plunder_vine", new ThornBlock(Settings.create().mapColor(MapColor.DARK_CRIMSON).hardness(1).ticksRandomly().sounds(BlockSoundGroup.WOOD).pistonBehavior(PistonBehavior.DESTROY), () -> UBlocks.PLUNDER_VINE_BUD));
Block PLUNDER_VINE_BUD = register("plunder_vine_bud", new ThornBudBlock(Settings.create().mapColor(MapColor.DARK_CRIMSON).hardness(1).nonOpaque().ticksRandomly().sounds(BlockSoundGroup.GRASS).pistonBehavior(PistonBehavior.DESTROY), PLUNDER_VINE.getDefaultState()));
+ CuringJokeBlock CURING_JOKE = register("curing_joke", new CuringJokeBlock(UEffects.BUTTER_FINGERS, 7, AbstractBlock.Settings.create().mapColor(MapColor.PALE_PURPLE).noCollision().breakInstantly().sounds(BlockSoundGroup.GRASS).offset(AbstractBlock.OffsetType.XZ).pistonBehavior(PistonBehavior.DESTROY)));
+ Block GOLD_ROOT = register("gold_root", new CarrotsBlock(AbstractBlock.Settings.create().mapColor(MapColor.GOLD).noCollision().ticksRandomly().breakInstantly().sounds(BlockSoundGroup.CROP).pistonBehavior(PistonBehavior.DESTROY)) {
+ @Override
+ protected ItemConvertible getSeedsItem() {
+ return Items.GOLDEN_CARROT;
+ }
+ });
Block CHITIN = register("chitin", new SnowyBlock(Settings.create().mapColor(MapColor.PALE_PURPLE).hardness(5).requiresTool().ticksRandomly().sounds(BlockSoundGroup.CORAL)), ItemGroups.NATURAL);
Block SURFACE_CHITIN = register("surface_chitin", new GrowableBlock(Settings.copy(CHITIN), () -> CHITIN), ItemGroups.NATURAL);
@@ -206,6 +225,8 @@ public interface UBlocks {
Block CRYSTAL_DOOR = register("crystal_door", new CrystalDoorBlock(Settings.copy(Blocks.IRON_DOOR), UWoodTypes.CRYSTAL), ItemGroups.FUNCTIONAL);
Block CLOUD_DOOR = register("cloud_door", new CloudDoorBlock(Settings.copy(CLOUD), CLOUD.getDefaultState(), UWoodTypes.CLOUD), ItemGroups.FUNCTIONAL);
+ EdibleBlock HAY_BLOCK = register("hay_block", new EdibleBlock(new Identifier("hay_block"), new Identifier("wheat"), true));
+
private static T register(String name, T item) {
return register(Unicopia.id(name), item);
}
@@ -229,10 +250,15 @@ public interface UBlocks {
if (block instanceof CloudLike || block instanceof SlimePustuleBlock || block instanceof PileBlock) {
SEMI_TRANSPARENT_BLOCKS.add(block);
}
+
return Registry.register(Registries.BLOCK, id, block);
}
static void bootstrap() {
+ if (FabricLoader.getInstance().isModLoaded("farmersdelight")) {
+ register("rice_block", new EdibleBlock(new Identifier("farmersdelight", "rice_bale"), new Identifier("farmersdelight", "rice_panicle"), true));
+ register("straw_block", new EdibleBlock(new Identifier("farmersdelight", "straw_bale"), new Identifier("farmersdelight", "straw"), true));
+ }
BlockEntityTypeSupportHelper.of(BlockEntityType.SIGN).addSupportedBlocks(PALM_SIGN, PALM_WALL_SIGN);
BlockEntityTypeSupportHelper.of(BlockEntityType.HANGING_SIGN).addSupportedBlocks(PALM_HANGING_SIGN, PALM_WALL_HANGING_SIGN);
@@ -240,21 +266,24 @@ public interface UBlocks {
StrippableBlockRegistry.register(PALM_LOG, STRIPPED_PALM_LOG);
StrippableBlockRegistry.register(ZAP_WOOD, STRIPPED_ZAP_WOOD);
StrippableBlockRegistry.register(PALM_WOOD, STRIPPED_PALM_WOOD);
- Collections.addAll(TRANSLUCENT_BLOCKS, WEATHER_VANE, CHITIN_SPIKES, PLUNDER_VINE, PLUNDER_VINE_BUD, CLAM_SHELL, SCALLOP_SHELL, TURRET_SHELL);
+ Collections.addAll(TRANSLUCENT_BLOCKS, WEATHER_VANE, CHITIN_SPIKES, PLUNDER_VINE, PLUNDER_VINE_BUD, CLAM_SHELL, SCALLOP_SHELL, TURRET_SHELL, CURING_JOKE);
TintedBlock.REGISTRY.add(PALM_LEAVES);
FlammableBlockRegistry.getDefaultInstance().add(GREEN_APPLE_LEAVES, 30, 60);
FlammableBlockRegistry.getDefaultInstance().add(SWEET_APPLE_LEAVES, 30, 60);
FlammableBlockRegistry.getDefaultInstance().add(SOUR_APPLE_LEAVES, 30, 60);
+ FlammableBlockRegistry.getDefaultInstance().add(GOLDEN_OAK_LEAVES, 60, 120);
FlammableBlockRegistry.getDefaultInstance().add(MANGO_LEAVES, 30, 60);
FlammableBlockRegistry.getDefaultInstance().add(PALM_LEAVES, 30, 60);
FlammableBlockRegistry.getDefaultInstance().add(PALM_LOG, 5, 5);
FlammableBlockRegistry.getDefaultInstance().add(PALM_WOOD, 5, 5);
+ FlammableBlockRegistry.getDefaultInstance().add(GOLDEN_OAK_LOG, 15, 15);
FlammableBlockRegistry.getDefaultInstance().add(STRIPPED_PALM_LOG, 5, 5);
FlammableBlockRegistry.getDefaultInstance().add(STRIPPED_PALM_WOOD, 5, 5);
FlammableBlockRegistry.getDefaultInstance().add(PALM_PLANKS, 5, 20);
FlammableBlockRegistry.getDefaultInstance().add(BANANAS, 5, 20);
UBlockEntities.bootstrap();
+ EdibleBlock.bootstrap();
}
}
diff --git a/src/main/java/com/minelittlepony/unicopia/block/UWoodTypes.java b/src/main/java/com/minelittlepony/unicopia/block/UWoodTypes.java
index 0a44586e..5f76ed97 100644
--- a/src/main/java/com/minelittlepony/unicopia/block/UWoodTypes.java
+++ b/src/main/java/com/minelittlepony/unicopia/block/UWoodTypes.java
@@ -15,6 +15,7 @@ import net.minecraft.util.Identifier;
public interface UWoodTypes {
WoodType PALM = register("palm");
+ WoodType GOLDEN_OAK = register("golden_oak");
RegistryKey PALM_BOAT_TYPE = TerraformBoatTypeRegistry.createKey(Unicopia.id("palm"));
BlockSetType CLOUD = new BlockSetTypeBuilder()
diff --git a/src/main/java/com/minelittlepony/unicopia/block/ZapAppleLeavesBlock.java b/src/main/java/com/minelittlepony/unicopia/block/ZapAppleLeavesBlock.java
index 4e83cbcf..e94f3475 100644
--- a/src/main/java/com/minelittlepony/unicopia/block/ZapAppleLeavesBlock.java
+++ b/src/main/java/com/minelittlepony/unicopia/block/ZapAppleLeavesBlock.java
@@ -4,11 +4,8 @@ import com.minelittlepony.unicopia.server.world.ZapAppleStageStore;
import net.minecraft.block.*;
import net.minecraft.item.ItemPlacementContext;
-import net.minecraft.server.world.ServerWorld;
import net.minecraft.state.StateManager;
import net.minecraft.state.property.*;
-import net.minecraft.util.math.BlockPos;
-import net.minecraft.world.World;
public class ZapAppleLeavesBlock extends BaseZapAppleLeavesBlock {
public static final EnumProperty STAGE = EnumProperty.of("stage", ZapAppleStageStore.Stage.class);
@@ -17,31 +14,13 @@ public class ZapAppleLeavesBlock extends BaseZapAppleLeavesBlock {
setDefaultState(getDefaultState().with(STAGE, ZapAppleStageStore.Stage.GREENING));
}
- @Override
- public void onBlockAdded(BlockState state, World world, BlockPos pos, BlockState oldState, boolean notify) {
- if (state.get(PERSISTENT)
- || oldState.isOf(state.getBlock())
- || oldState.isOf(UBlocks.ZAP_LEAVES)
- || oldState.isOf(UBlocks.FLOWERING_ZAP_LEAVES)
- || oldState.isOf(UBlocks.ZAP_LEAVES_PLACEHOLDER)
- || !(world instanceof ServerWorld sw)) {
- return;
- }
-
- ZapAppleStageStore store = ZapAppleStageStore.get(sw);
- ZapAppleStageStore.Stage currentStage = store.getStage();
- if (currentStage != getStage(state)) {
- world.setBlockState(pos, currentStage.getNewState(state));
- }
- }
-
@Override
public BlockState getPlacementState(ItemPlacementContext ctx) {
return super.getPlacementState(ctx).with(STAGE, ZapAppleStageStore.Stage.GREENING);
}
@Override
- protected ZapAppleStageStore.Stage getStage(BlockState state) {
+ public ZapAppleStageStore.Stage getStage(BlockState state) {
return state.get(STAGE);
}
diff --git a/src/main/java/com/minelittlepony/unicopia/block/ZapAppleLeavesPlaceholderBlock.java b/src/main/java/com/minelittlepony/unicopia/block/ZapAppleLeavesPlaceholderBlock.java
index 5b6a7cdc..40fa9fc2 100644
--- a/src/main/java/com/minelittlepony/unicopia/block/ZapAppleLeavesPlaceholderBlock.java
+++ b/src/main/java/com/minelittlepony/unicopia/block/ZapAppleLeavesPlaceholderBlock.java
@@ -1,51 +1,34 @@
package com.minelittlepony.unicopia.block;
import com.minelittlepony.unicopia.server.world.ZapAppleStageStore;
+import com.minelittlepony.unicopia.server.world.ZapAppleStageStore.Stage;
import net.minecraft.block.*;
import net.minecraft.server.world.ServerWorld;
import net.minecraft.util.math.*;
import net.minecraft.util.math.random.Random;
-import net.minecraft.world.WorldAccess;
+import net.minecraft.world.World;
-public class ZapAppleLeavesPlaceholderBlock extends AirBlock {
+public class ZapAppleLeavesPlaceholderBlock extends AirBlock implements ZapStagedBlock {
ZapAppleLeavesPlaceholderBlock() {
super(Settings.create().replaceable().noCollision().dropsNothing().air());
}
@Override
- public boolean hasRandomTicks(BlockState state) {
- return true;
+ public Stage getStage(BlockState state) {
+ return ZapAppleStageStore.Stage.HIBERNATING;
}
@Override
- public BlockState getStateForNeighborUpdate(BlockState state, Direction direction, BlockState neighborState, WorldAccess world, BlockPos pos, BlockPos neighborPos) {
-
- if (world instanceof ServerWorld sw) {
- ZapAppleStageStore store = ZapAppleStageStore.get(sw);
- ZapAppleStageStore.Stage currentStage = store.getStage();
- if (currentStage != ZapAppleStageStore.Stage.HIBERNATING) {
- return currentStage.getNewState(state);
- }
- }
-
- return state;
+ public void onBlockAdded(BlockState state, World world, BlockPos pos, BlockState oldState, boolean notify) {
+ updateStage(state, world, pos);
}
@Deprecated
@Override
public void scheduledTick(BlockState state, ServerWorld world, BlockPos pos, Random random) {
super.scheduledTick(state, world, pos, random);
-
- ZapAppleStageStore store = ZapAppleStageStore.get(world);
- ZapAppleStageStore.Stage newStage = store.getStage();
- if (!world.isDay() && ZapAppleStageStore.Stage.HIBERNATING.mustChangeIntoInstantly(newStage)) {
- state = newStage.getNewState(state);
- world.setBlockState(pos, state);
- BaseZapAppleLeavesBlock.onStageChanged(store, newStage, world, state, pos, random);
- }
-
- world.scheduleBlockTick(pos, state.getBlock(), 1);
+ tryAdvanceStage(state, world, pos, random);
}
}
diff --git a/src/main/java/com/minelittlepony/unicopia/block/ZapStagedBlock.java b/src/main/java/com/minelittlepony/unicopia/block/ZapStagedBlock.java
new file mode 100644
index 00000000..5e3f41d4
--- /dev/null
+++ b/src/main/java/com/minelittlepony/unicopia/block/ZapStagedBlock.java
@@ -0,0 +1,87 @@
+package com.minelittlepony.unicopia.block;
+
+import com.minelittlepony.unicopia.server.world.ZapAppleStageStore;
+
+import net.minecraft.block.Block;
+import net.minecraft.block.BlockState;
+import net.minecraft.block.Blocks;
+import net.minecraft.server.world.ServerWorld;
+import net.minecraft.util.math.BlockPos;
+import net.minecraft.util.math.random.Random;
+import net.minecraft.world.World;
+
+public interface ZapStagedBlock {
+ ZapAppleStageStore.Stage getStage(BlockState state);
+
+ default void updateStage(BlockState state, World world, BlockPos pos) {
+ if (!(world instanceof ServerWorld sw)) {
+ return;
+ }
+ ZapAppleStageStore.Stage currentStage = ZapAppleStageStore.get(sw).getStage();
+ if (currentStage != getStage(state)) {
+ state = getState(currentStage);
+ world.setBlockState(pos, state);
+ }
+ world.scheduleBlockTick(pos, state.getBlock(), 1);
+ }
+
+ default void tryAdvanceStage(BlockState state, ServerWorld world, BlockPos pos, Random random) {
+ ZapAppleStageStore store = ZapAppleStageStore.get(world);
+ ZapAppleStageStore.Stage currentStage = store.getStage();
+ if (!world.isDay() && currentStage != getStage(state)) {
+ int transitionRate = getTransitionRate(currentStage);
+ if (transitionRate == 0 || random.nextInt(transitionRate) == 0) {
+ state = getState(currentStage);
+ world.setBlockState(pos, state);
+ onStageChanged(store, currentStage, world, state, pos, random);
+ }
+ }
+ world.scheduleBlockTick(pos, state.getBlock(), 1);
+ }
+
+ default int getTransitionRate(ZapAppleStageStore.Stage stage) {
+ if (stage == ZapAppleStageStore.Stage.HIBERNATING || stage == ZapAppleStageStore.Stage.GREENING) {
+ return 10;
+ }
+ return 2500;
+ }
+
+ default BlockState getState(ZapAppleStageStore.Stage stage) {
+ if (stage == ZapAppleStageStore.Stage.HIBERNATING) {
+ return UBlocks.ZAP_LEAVES_PLACEHOLDER.getDefaultState();
+ }
+ if (stage == ZapAppleStageStore.Stage.FLOWERING) {
+ return UBlocks.FLOWERING_ZAP_LEAVES.getDefaultState();
+ }
+ return UBlocks.ZAP_LEAVES.getDefaultState().with(ZapAppleLeavesBlock.STAGE, stage);
+ }
+
+ private static void onStageChanged(ZapAppleStageStore store, ZapAppleStageStore.Stage stage, ServerWorld world, BlockState state, BlockPos pos, Random random) {
+ boolean mustFruit = Random.create(state.getRenderingSeed(pos)).nextInt(5) < 2;
+ BlockState below = world.getBlockState(pos.down());
+
+ if (world.isAir(pos.down())) {
+ if (stage == ZapAppleStageStore.Stage.FRUITING && mustFruit) {
+ world.setBlockState(pos.down(), UBlocks.ZAP_BULB.getDefaultState(), Block.NOTIFY_ALL);
+ store.triggerLightningStrike(pos);
+ }
+ }
+
+ if (stage != ZapAppleStageStore.Stage.HIBERNATING && world.getRandom().nextInt(10) == 0) {
+ store.triggerLightningStrike(pos);
+ }
+
+ if (stage == ZapAppleStageStore.Stage.RIPE) {
+ if (below.isOf(UBlocks.ZAP_BULB)) {
+ world.setBlockState(pos.down(), UBlocks.ZAP_APPLE.getDefaultState(), Block.NOTIFY_ALL);
+ store.playMoonEffect(pos);
+ }
+ }
+
+ if (mustFruit && stage == ZapAppleStageStore.Stage.HIBERNATING) {
+ if (below.isOf(UBlocks.ZAP_APPLE) || below.isOf(UBlocks.ZAP_BULB)) {
+ world.setBlockState(pos.down(), Blocks.AIR.getDefaultState());
+ }
+ }
+ }
+}
diff --git a/src/main/java/com/minelittlepony/unicopia/block/cloud/PoreousCloudBlock.java b/src/main/java/com/minelittlepony/unicopia/block/cloud/PoreousCloudBlock.java
index 5e17ca06..15303184 100644
--- a/src/main/java/com/minelittlepony/unicopia/block/cloud/PoreousCloudBlock.java
+++ b/src/main/java/com/minelittlepony/unicopia/block/cloud/PoreousCloudBlock.java
@@ -4,6 +4,8 @@ import java.util.function.Supplier;
import org.jetbrains.annotations.Nullable;
+import com.minelittlepony.unicopia.block.state.StateUtil;
+
import net.minecraft.block.BlockState;
import net.minecraft.server.world.ServerWorld;
import net.minecraft.util.math.BlockPos;
@@ -22,7 +24,7 @@ public class PoreousCloudBlock extends CloudBlock implements Soakable {
@Override
public BlockState getStateWithMoisture(BlockState state, int moisture) {
if (moisture <= 0) {
- return Soakable.copyProperties(state, getDefaultState());
+ return StateUtil.copyState(state, getDefaultState());
}
return soggyBlock == null ? null : soggyBlock.get().getStateWithMoisture(state, moisture);
}
diff --git a/src/main/java/com/minelittlepony/unicopia/block/cloud/PoreousCloudStairsBlock.java b/src/main/java/com/minelittlepony/unicopia/block/cloud/PoreousCloudStairsBlock.java
index 6733ee2d..3ee498a1 100644
--- a/src/main/java/com/minelittlepony/unicopia/block/cloud/PoreousCloudStairsBlock.java
+++ b/src/main/java/com/minelittlepony/unicopia/block/cloud/PoreousCloudStairsBlock.java
@@ -4,6 +4,8 @@ import java.util.function.Supplier;
import org.jetbrains.annotations.Nullable;
+import com.minelittlepony.unicopia.block.state.StateUtil;
+
import net.minecraft.block.BlockState;
public class PoreousCloudStairsBlock extends CloudStairsBlock implements Soakable {
@@ -19,7 +21,7 @@ public class PoreousCloudStairsBlock extends CloudStairsBlock implements Soakabl
@Override
public BlockState getStateWithMoisture(BlockState state, int moisture) {
if (moisture <= 0) {
- return Soakable.copyProperties(state, getDefaultState());
+ return StateUtil.copyState(state, getDefaultState());
}
return soggyBlock.get().getStateWithMoisture(state, moisture);
}
diff --git a/src/main/java/com/minelittlepony/unicopia/block/cloud/Soakable.java b/src/main/java/com/minelittlepony/unicopia/block/cloud/Soakable.java
index e800bbf8..94ce19e8 100644
--- a/src/main/java/com/minelittlepony/unicopia/block/cloud/Soakable.java
+++ b/src/main/java/com/minelittlepony/unicopia/block/cloud/Soakable.java
@@ -15,7 +15,6 @@ import net.minecraft.server.world.ServerWorld;
import net.minecraft.sound.SoundCategory;
import net.minecraft.sound.SoundEvents;
import net.minecraft.state.property.IntProperty;
-import net.minecraft.state.property.Property;
import net.minecraft.util.ActionResult;
import net.minecraft.util.Hand;
import net.minecraft.util.Util;
@@ -105,15 +104,4 @@ public interface Soakable {
world.setBlockState(pos, soakable.getStateWithMoisture(state, newMoisture));
world.playSound(null, pos, SoundEvents.ENTITY_SALMON_FLOP, SoundCategory.BLOCKS, 1, (float)world.random.nextTriangular(0.5, 0.3F));
}
-
- @Nullable
- @SuppressWarnings({ "rawtypes", "unchecked" })
- static BlockState copyProperties(BlockState from, @Nullable BlockState to) {
- if (to != null) {
- for (Property property : from.getProperties()) {
- to = to.withIfExists(property, from.get(property));
- }
- }
- return to;
- }
}
diff --git a/src/main/java/com/minelittlepony/unicopia/block/cloud/SoggyCloudBlock.java b/src/main/java/com/minelittlepony/unicopia/block/cloud/SoggyCloudBlock.java
index e508b96a..5f769024 100644
--- a/src/main/java/com/minelittlepony/unicopia/block/cloud/SoggyCloudBlock.java
+++ b/src/main/java/com/minelittlepony/unicopia/block/cloud/SoggyCloudBlock.java
@@ -4,6 +4,8 @@ import java.util.function.Supplier;
import org.jetbrains.annotations.Nullable;
+import com.minelittlepony.unicopia.block.state.StateUtil;
+
import net.minecraft.block.Block;
import net.minecraft.block.BlockState;
import net.minecraft.entity.player.PlayerEntity;
@@ -43,9 +45,9 @@ public class SoggyCloudBlock extends CloudBlock implements Soakable {
@Override
public BlockState getStateWithMoisture(BlockState state, int moisture) {
if (moisture <= 0) {
- return Soakable.copyProperties(state, dryBlock.get().getDefaultState());
+ return StateUtil.copyState(state, dryBlock.get().getDefaultState());
}
- return Soakable.copyProperties(state, getDefaultState()).with(MOISTURE, moisture);
+ return StateUtil.copyState(state, getDefaultState()).with(MOISTURE, moisture);
}
@Override
diff --git a/src/main/java/com/minelittlepony/unicopia/block/cloud/SoggyCloudSlabBlock.java b/src/main/java/com/minelittlepony/unicopia/block/cloud/SoggyCloudSlabBlock.java
index d074b2cb..7595c2b1 100644
--- a/src/main/java/com/minelittlepony/unicopia/block/cloud/SoggyCloudSlabBlock.java
+++ b/src/main/java/com/minelittlepony/unicopia/block/cloud/SoggyCloudSlabBlock.java
@@ -4,6 +4,8 @@ import java.util.function.Supplier;
import org.jetbrains.annotations.Nullable;
+import com.minelittlepony.unicopia.block.state.StateUtil;
+
import net.minecraft.block.Block;
import net.minecraft.block.BlockState;
import net.minecraft.entity.player.PlayerEntity;
@@ -43,9 +45,9 @@ public class SoggyCloudSlabBlock extends CloudSlabBlock {
@Override
public BlockState getStateWithMoisture(BlockState state, int moisture) {
if (moisture <= 0) {
- return Soakable.copyProperties(state, dryBlock.get().getDefaultState());
+ return StateUtil.copyState(state, dryBlock.get().getDefaultState());
}
- return Soakable.copyProperties(state, getDefaultState()).with(MOISTURE, moisture);
+ return StateUtil.copyState(state, getDefaultState()).with(MOISTURE, moisture);
}
@Override
diff --git a/src/main/java/com/minelittlepony/unicopia/block/cloud/SoggyCloudStairsBlock.java b/src/main/java/com/minelittlepony/unicopia/block/cloud/SoggyCloudStairsBlock.java
index 495c0f37..9e51e398 100644
--- a/src/main/java/com/minelittlepony/unicopia/block/cloud/SoggyCloudStairsBlock.java
+++ b/src/main/java/com/minelittlepony/unicopia/block/cloud/SoggyCloudStairsBlock.java
@@ -4,6 +4,8 @@ import java.util.function.Supplier;
import org.jetbrains.annotations.Nullable;
+import com.minelittlepony.unicopia.block.state.StateUtil;
+
import net.minecraft.block.Block;
import net.minecraft.block.BlockState;
import net.minecraft.item.ItemStack;
@@ -36,8 +38,8 @@ public class SoggyCloudStairsBlock extends CloudStairsBlock implements Soakable
@Override
public BlockState getStateWithMoisture(BlockState state, int moisture) {
if (moisture <= 0) {
- return Soakable.copyProperties(state, dryBlock.get().getDefaultState());
+ return StateUtil.copyState(state, dryBlock.get().getDefaultState());
}
- return Soakable.copyProperties(state, getDefaultState()).with(MOISTURE, moisture);
+ return StateUtil.copyState(state, getDefaultState()).with(MOISTURE, moisture);
}
}
diff --git a/src/main/java/com/minelittlepony/unicopia/block/state/StateChange.java b/src/main/java/com/minelittlepony/unicopia/block/state/StateChange.java
index 72e7b9e4..972eef4f 100644
--- a/src/main/java/com/minelittlepony/unicopia/block/state/StateChange.java
+++ b/src/main/java/com/minelittlepony/unicopia/block/state/StateChange.java
@@ -11,7 +11,6 @@ import com.minelittlepony.unicopia.Unicopia;
import net.minecraft.block.Block;
import net.minecraft.block.BlockState;
-import net.minecraft.state.property.Property;
import net.minecraft.util.Identifier;
import net.minecraft.util.JsonHelper;
import net.minecraft.registry.Registries;
@@ -50,7 +49,7 @@ public abstract class StateChange {
return state;
}
return Registries.BLOCK.getOrEmpty(id).map(Block::getDefaultState)
- .map(newState -> merge(newState, state))
+ .map(newState -> StateUtil.copyState(state, newState))
.orElse(state);
}
};
@@ -101,17 +100,4 @@ public abstract class StateChange {
return serializer.apply(json);
}).orElseThrow(() -> new IllegalArgumentException("Invalid action " + action));
}
-
- private static BlockState merge(BlockState into, BlockState from) {
- for (var property : from.getProperties()) {
- if (into.contains(property)) {
- into = copy(into, from, property);
- }
- }
- return into;
- }
-
- private static > BlockState copy(BlockState to, BlockState from, Property property) {
- return to.with(property, from.get(property));
- }
}
diff --git a/src/main/java/com/minelittlepony/unicopia/block/state/StateUtil.java b/src/main/java/com/minelittlepony/unicopia/block/state/StateUtil.java
new file mode 100644
index 00000000..42334368
--- /dev/null
+++ b/src/main/java/com/minelittlepony/unicopia/block/state/StateUtil.java
@@ -0,0 +1,19 @@
+package com.minelittlepony.unicopia.block.state;
+
+import org.jetbrains.annotations.Nullable;
+
+import net.minecraft.block.BlockState;
+import net.minecraft.state.property.Property;
+
+public interface StateUtil {
+ @SuppressWarnings({ "unchecked", "rawtypes" })
+ static BlockState copyState(BlockState from, @Nullable BlockState to) {
+ if (to == null) {
+ return to;
+ }
+ for (var property : from.getProperties()) {
+ to = to.withIfExists((Property)property, from.get(property));
+ }
+ return to;
+ }
+}
diff --git a/src/main/java/com/minelittlepony/unicopia/client/ClientInteractionManager.java b/src/main/java/com/minelittlepony/unicopia/client/ClientInteractionManager.java
index 0be0da61..837b65fd 100644
--- a/src/main/java/com/minelittlepony/unicopia/client/ClientInteractionManager.java
+++ b/src/main/java/com/minelittlepony/unicopia/client/ClientInteractionManager.java
@@ -2,7 +2,7 @@ package com.minelittlepony.unicopia.client;
import java.lang.ref.WeakReference;
import java.util.Map;
-import java.util.Optional;
+import java.util.UUID;
import java.util.function.Predicate;
import java.util.function.Supplier;
@@ -13,19 +13,20 @@ import com.minelittlepony.unicopia.EquinePredicates;
import com.minelittlepony.unicopia.FlightType;
import com.minelittlepony.unicopia.InteractionManager;
import com.minelittlepony.unicopia.USounds;
-import com.minelittlepony.unicopia.ability.magic.CasterView;
import com.minelittlepony.unicopia.client.gui.DismissSpellScreen;
import com.minelittlepony.unicopia.client.gui.spellbook.ClientChapters;
+import com.minelittlepony.unicopia.client.particle.ClientBoundParticleSpawner;
import com.minelittlepony.unicopia.client.sound.*;
import com.minelittlepony.unicopia.entity.player.PlayerPhysics;
import com.minelittlepony.unicopia.entity.player.Pony;
import com.minelittlepony.unicopia.entity.player.dummy.DummyClientPlayerEntity;
-import com.minelittlepony.unicopia.server.world.Ether;
+import com.minelittlepony.unicopia.particle.ParticleSpawner;
import com.mojang.authlib.GameProfile;
import it.unimi.dsi.fastutil.ints.Int2ObjectMap;
import it.unimi.dsi.fastutil.ints.Int2ObjectOpenHashMap;
import net.minecraft.client.MinecraftClient;
+import net.minecraft.client.network.ClientPlayerEntity;
import net.minecraft.client.sound.AggressiveBeeSoundInstance;
import net.minecraft.client.sound.MovingMinecartSoundInstance;
import net.minecraft.client.sound.PassiveBeeSoundInstance;
@@ -37,32 +38,21 @@ import net.minecraft.entity.passive.BeeEntity;
import net.minecraft.entity.player.PlayerEntity;
import net.minecraft.entity.vehicle.AbstractMinecartEntity;
import net.minecraft.network.PacketByteBuf;
-import net.minecraft.server.world.ServerWorld;
+import net.minecraft.network.packet.c2s.play.PlayerMoveC2SPacket;
import net.minecraft.sound.SoundCategory;
import net.minecraft.util.Identifier;
import net.minecraft.util.math.random.Random;
-import net.minecraft.world.BlockView;
import net.minecraft.world.World;
public class ClientInteractionManager extends InteractionManager {
private final MinecraftClient client = MinecraftClient.getInstance();
- private final Optional clientWorld = Optional.of(() -> MinecraftClient.getInstance().world);
-
private final Int2ObjectMap> playingSounds = new Int2ObjectOpenHashMap<>();
- @Override
- public Optional getCasterView(BlockView view) {
- if (view instanceof ServerWorld world) {
- return Optional.of(Ether.get(world));
- }
- return clientWorld;
- }
-
@Override
public Map readChapters(PacketByteBuf buffer) {
- return buffer.readMap(PacketByteBuf::readIdentifier, ClientChapters::loadChapter);
+ return buffer.readMap(PacketByteBuf::readIdentifier, ClientChapters::loadChapter);
}
@Override
@@ -159,4 +149,16 @@ public class ClientInteractionManager extends InteractionManager {
public int getViewMode() {
return client.options.getPerspective().ordinal();
}
+
+ @Override
+ public ParticleSpawner createBoundParticle(UUID id) {
+ return new ClientBoundParticleSpawner(id);
+ }
+
+ @Override
+ public void sendPlayerLookAngles(PlayerEntity player) {
+ if (player instanceof ClientPlayerEntity c) {
+ c.networkHandler.sendPacket(new PlayerMoveC2SPacket.LookAndOnGround(player.getYaw(), player.getPitch(), player.isOnGround()));
+ }
+ }
}
diff --git a/src/main/java/com/minelittlepony/unicopia/client/URenderers.java b/src/main/java/com/minelittlepony/unicopia/client/URenderers.java
index 5bf94a22..ab09e503 100644
--- a/src/main/java/com/minelittlepony/unicopia/client/URenderers.java
+++ b/src/main/java/com/minelittlepony/unicopia/client/URenderers.java
@@ -10,6 +10,8 @@ import com.minelittlepony.unicopia.block.cloud.CloudChestBlock;
import com.minelittlepony.unicopia.client.particle.ChangelingMagicParticle;
import com.minelittlepony.unicopia.client.particle.CloudsEscapingParticle;
import com.minelittlepony.unicopia.client.particle.DiskParticle;
+import com.minelittlepony.unicopia.client.particle.DustCloudParticle;
+import com.minelittlepony.unicopia.client.particle.FloatingBubbleParticle;
import com.minelittlepony.unicopia.client.particle.GroundPoundParticle;
import com.minelittlepony.unicopia.client.particle.HealthDrainParticle;
import com.minelittlepony.unicopia.client.particle.LightningBoltParticle;
@@ -22,6 +24,7 @@ import com.minelittlepony.unicopia.client.particle.ShockwaveParticle;
import com.minelittlepony.unicopia.client.particle.SphereParticle;
import com.minelittlepony.unicopia.client.render.*;
import com.minelittlepony.unicopia.client.render.entity.*;
+import com.minelittlepony.unicopia.client.render.shader.UShaders;
import com.minelittlepony.unicopia.client.render.spell.SpellRendererFactory;
import com.minelittlepony.unicopia.entity.mob.UEntities;
import com.minelittlepony.unicopia.item.ChameleonItem;
@@ -59,12 +62,14 @@ import net.minecraft.util.Identifier;
import net.minecraft.util.math.BlockPos;
import net.minecraft.world.BlockRenderView;
+@SuppressWarnings("deprecation")
public interface URenderers {
BlockEntity CHEST_RENDER_ENTITY = new CloudChestBlock.TileData(BlockPos.ORIGIN, UBlocks.CLOUD_CHEST.getDefaultState());
static void bootstrap() {
ParticleFactoryRegistry.getInstance().register(UParticles.UNICORN_MAGIC, createFactory(MagicParticle::new));
ParticleFactoryRegistry.getInstance().register(UParticles.CHANGELING_MAGIC, createFactory(ChangelingMagicParticle::new));
+ ParticleFactoryRegistry.getInstance().register(UParticles.BUBBLE, createFactory(FloatingBubbleParticle::new));
ParticleFactoryRegistry.getInstance().register(UParticles.RAIN_DROPS, createFactory(RaindropsParticle::new));
ParticleFactoryRegistry.getInstance().register(UParticles.HEALTH_DRAIN, createFactory(HealthDrainParticle::create));
ParticleFactoryRegistry.getInstance().register(UParticles.RAINBOOM_RING, RainboomParticle::new);
@@ -76,6 +81,7 @@ public interface URenderers {
ParticleFactoryRegistry.getInstance().register(UParticles.GROUND_POUND, GroundPoundParticle::new);
ParticleFactoryRegistry.getInstance().register(UParticles.CLOUDS_ESCAPING, CloudsEscapingParticle::new);
ParticleFactoryRegistry.getInstance().register(UParticles.LIGHTNING_BOLT, LightningBoltParticle::new);
+ ParticleFactoryRegistry.getInstance().register(UParticles.DUST_CLOUD, DustCloudParticle::new);
AccessoryFeatureRenderer.register(
BraceletFeatureRenderer::new, AmuletFeatureRenderer::new, GlassesFeatureRenderer::new,
@@ -97,6 +103,8 @@ public interface URenderers {
EntityRendererRegistry.register(UEntities.AIR_BALLOON, AirBalloonEntityRenderer::new);
EntityRendererRegistry.register(UEntities.FRIENDLY_CREEPER, FriendlyCreeperEntityRenderer::new);
EntityRendererRegistry.register(UEntities.LOOT_BUG, LootBugEntityRenderer::new);
+ EntityRendererRegistry.register(UEntities.TENTACLE, TentacleEntityRenderer::new);
+ EntityRendererRegistry.register(UEntities.IGNOMINIOUS_BULB, IgnominiousBulbEntityRenderer::new);
BlockEntityRendererFactories.register(UBlockEntities.WEATHER_VANE, WeatherVaneBlockEntityRenderer::new);
BlockEntityRendererFactories.register(UBlockEntities.FANCY_BED, CloudBedBlockEntityRenderer::new);
@@ -108,6 +116,7 @@ public interface URenderers {
PolearmRenderer.register(UItems.WOODEN_POLEARM, UItems.STONE_POLEARM, UItems.IRON_POLEARM, UItems.GOLDEN_POLEARM, UItems.DIAMOND_POLEARM, UItems.NETHERITE_POLEARM);
ModelPredicateProviderRegistry.register(UItems.GEMSTONE, new Identifier("affinity"), (stack, world, entity, seed) -> EnchantableItem.isEnchanted(stack) ? EnchantableItem.getSpellKey(stack).getAffinity().getAlignment() : 0);
ModelPredicateProviderRegistry.register(UItems.ROCK_CANDY, new Identifier("count"), (stack, world, entity, seed) -> stack.getCount() / (float)stack.getMaxCount());
+ ModelPredicateProviderRegistry.register(Unicopia.id("zap_cycle"), (stack, world, entity, seed) -> UnicopiaClient.getInstance().getZapStageDelta());
ColorProviderRegistry.BLOCK.register(URenderers::getTintedBlockColor, TintedBlock.REGISTRY.stream().toArray(Block[]::new));
ColorProviderRegistry.ITEM.register((stack, i) -> getTintedBlockColor(Block.getBlockFromItem(stack.getItem()).getDefaultState(), null, null, i), TintedBlock.REGISTRY.stream().map(Block::asItem).filter(i -> i != Items.AIR).toArray(Item[]::new));
@@ -123,6 +132,7 @@ public interface URenderers {
TerraformBoatClientHelper.registerModelLayers(Unicopia.id("palm"), false);
SpellRendererFactory.bootstrap();
+ UShaders.bootstrap();
}
private static void register(DynamicItemRenderer renderer, ItemConvertible...items) {
diff --git a/src/main/java/com/minelittlepony/unicopia/client/UnicopiaClient.java b/src/main/java/com/minelittlepony/unicopia/client/UnicopiaClient.java
index 32b01a8e..9ea6a700 100644
--- a/src/main/java/com/minelittlepony/unicopia/client/UnicopiaClient.java
+++ b/src/main/java/com/minelittlepony/unicopia/client/UnicopiaClient.java
@@ -22,6 +22,7 @@ import com.minelittlepony.unicopia.entity.player.PlayerCamera;
import com.minelittlepony.unicopia.entity.player.Pony;
import com.minelittlepony.unicopia.network.handler.ClientNetworkHandlerImpl;
import com.minelittlepony.unicopia.server.world.WeatherConditions;
+import com.minelittlepony.unicopia.server.world.ZapAppleStageStore;
import com.minelittlepony.unicopia.util.Lerp;
import net.fabricmc.api.ClientModInitializer;
@@ -53,6 +54,9 @@ public class UnicopiaClient implements ClientModInitializer {
public final Lerp tangentalSkyAngle = new Lerp(0, true);
public final Lerp skyAngle = new Lerp(0, true);
+ private ZapAppleStageStore.Stage zapAppleStage = ZapAppleStageStore.Stage.HIBERNATING;
+ private long zapStageTime;
+
public static Optional getCamera() {
PlayerEntity player = MinecraftClient.getInstance().player;
@@ -84,6 +88,15 @@ public class UnicopiaClient implements ClientModInitializer {
instance = this;
}
+ public void setZapAppleStage(ZapAppleStageStore.Stage stage, long delta) {
+ zapAppleStage = stage;
+ zapStageTime = delta;
+ }
+
+ public float getZapStageDelta() {
+ return zapAppleStage.getCycleProgress(zapStageTime);
+ }
+
public float getSkyAngleDelta(float tickDelta) {
if (MinecraftClient.getInstance().world == null) {
return 0;
@@ -135,6 +148,8 @@ public class UnicopiaClient implements ClientModInitializer {
world.setRainGradient(gradient);
world.setThunderGradient(gradient);
}
+
+ zapStageTime++;
}
private Float getTargetRainGradient(ClientWorld world, BlockPos pos, float tickDelta) {
diff --git a/src/main/java/com/minelittlepony/unicopia/client/gui/MagicText.java b/src/main/java/com/minelittlepony/unicopia/client/gui/MagicText.java
new file mode 100644
index 00000000..4387217f
--- /dev/null
+++ b/src/main/java/com/minelittlepony/unicopia/client/gui/MagicText.java
@@ -0,0 +1,16 @@
+package com.minelittlepony.unicopia.client.gui;
+
+import net.minecraft.client.MinecraftClient;
+import net.minecraft.util.math.MathHelper;
+
+public interface MagicText {
+ static int getColor() {
+ MinecraftClient client = MinecraftClient.getInstance();
+ float ticks = client.player.age + client.getTickDelta();
+
+ float sin = (MathHelper.sin(ticks / 10F) + 1) * 155 * 0.25F;
+ float cos = (MathHelper.cos((ticks + 10) / 10F) + 1) * 155 * 0.25F;
+
+ return (int)(sin + cos);
+ }
+}
diff --git a/src/main/java/com/minelittlepony/unicopia/client/gui/ParagraphWrappingVisitor.java b/src/main/java/com/minelittlepony/unicopia/client/gui/ParagraphWrappingVisitor.java
index 0a947e9c..66f38d23 100644
--- a/src/main/java/com/minelittlepony/unicopia/client/gui/ParagraphWrappingVisitor.java
+++ b/src/main/java/com/minelittlepony/unicopia/client/gui/ParagraphWrappingVisitor.java
@@ -2,7 +2,6 @@ package com.minelittlepony.unicopia.client.gui;
import java.util.*;
import java.util.function.BiConsumer;
-
import it.unimi.dsi.fastutil.ints.Int2IntFunction;
import net.minecraft.client.MinecraftClient;
import net.minecraft.client.font.TextHandler;
@@ -31,12 +30,18 @@ public class ParagraphWrappingVisitor implements StyledVisitor