diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index b5e6673..eba985c 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -5,15 +5,15 @@ jobs: build: strategy: matrix: - os: [ubuntu-20.04] + os: [ubuntu-latest] runs-on: ${{ matrix.os }} steps: - name: checkout repository - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: validate gradle wrapper uses: gradle/wrapper-validation-action@v1 - - name: setup jdk ${{ matrix.java }} - uses: actions/setup-java@v3 + - name: setup jdk + uses: actions/setup-java@v4 with: distribution: 'adopt' java-version: 21 @@ -22,7 +22,7 @@ jobs: - name: build run: ./gradlew build - name: capture build artifacts - uses: actions/upload-artifact@v3 + uses: actions/upload-artifact@v4 with: name: Artifacts path: build/libs/ diff --git a/README.md b/README.md index 9ee7fbc..faa777f 100644 --- a/README.md +++ b/README.md @@ -12,7 +12,7 @@ Improve horses by removing stupid stuff and adding useful stuff.
Horses Sit in Rafts - Rafts only because its impossible to break boats when they are in a boat. + Rafts only because its impossible to break boats when they are in a boat. Works with horses, donkeys, mules, zombie horses, skeleton horses, camels and llamas. ![player sitting in a bamboo raft in a river with a horse](https://cdn.modrinth.com/data/cached_images/d2faef4b56e873ee8a20816b5276e52aa1b20bbc.png)
@@ -71,6 +71,8 @@ Improve horses by removing stupid stuff and adding useful stuff.
Fade Horse as you look down + The same applies to donkeys, mules, zombie horses, skeleton horses, camels and llamas. + ![going down a hill into a cave, looking down through horse for better navigation](https://i.imgur.com/kwtmFVd.gif)
@@ -78,6 +80,7 @@ Improve horses by removing stupid stuff and adding useful stuff. Horse Head Pitch offset Option to lower horse's head down up to 45 degrees to improve visibility. + Works with horses, donkeys, mules, zombie horses and skeleton horses. ![comparison of 0 vs 45 degree offset](https://user-images.githubusercontent.com/37855219/163890939-87f3b255-176a-49df-ad4d-ec5b41a3b54a.png) @@ -85,7 +88,10 @@ Improve horses by removing stupid stuff and adding useful stuff.
Horses named `jeb_` become rainbow + The same can be applied to donkeys, mules, camels, llamas, zombie horses and skeleton horses. + ![Jeb_ rainbow horse, like the sheep easter egg](https://i.imgur.com/QTk8w33.gif) +
diff --git a/build.gradle b/build.gradle index 1ca9818..530a4dc 100644 --- a/build.gradle +++ b/build.gradle @@ -1,19 +1,15 @@ plugins { - id 'fabric-loom' version '1.6-SNAPSHOT' + id 'fabric-loom' version '1.9-SNAPSHOT' id 'maven-publish' } -sourceCompatibility = JavaVersion.VERSION_17 -targetCompatibility = JavaVersion.VERSION_17 +sourceCompatibility = JavaVersion.VERSION_21 +targetCompatibility = JavaVersion.VERSION_21 archivesBaseName = project.archives_base_name + "-" + project.minecraft_version version = project.mod_version group = project.maven_group -loom { - accessWidenerPath = file("src/main/resources/horsebuff.accesswidener") -} - configurations { modIncludeImplementation @@ -66,7 +62,7 @@ tasks.withType(JavaCompile).configureEach { // If Javadoc is generated, this must be specified in that task too. it.options.encoding = "UTF-8" - it.options.release = 17 + it.options.release = 21 } java { // Loom will automatically attach sourcesJar to a RemapSourcesJar task and to the "build" task diff --git a/gradle.properties b/gradle.properties index bdf1f8d..8b5388c 100644 --- a/gradle.properties +++ b/gradle.properties @@ -3,15 +3,15 @@ org.gradle.jvmargs=-Xmx1G # Fabric Properties # check these on https://fabricmc.net/develop/ -minecraft_version=1.21 -yarn_mappings=1.21+build.2 -loader_version=0.15.11 +minecraft_version=1.21.3 +yarn_mappings=1.21.3+build.2 +loader_version=0.16.9 # Mod Properties -mod_version=2.1.8 +mod_version=2.2.0 maven_group=com.HorseBuff archives_base_name=HorseBuff # Dependencies -fabric_version=0.100.3+1.21 -cloth_config_version=15.0.127 -mod_menu_version=11.0.1 -mixinextras_version=0.3.5 \ No newline at end of file +fabric_version=0.112.1+1.21.3 +cloth_config_version=16.0.143 +mod_menu_version=12.0.0 +mixinextras_version=0.4.1 \ No newline at end of file diff --git a/gradle/wrapper/gradle-wrapper.properties b/gradle/wrapper/gradle-wrapper.properties index 48c0a02..e48eca5 100644 --- a/gradle/wrapper/gradle-wrapper.properties +++ b/gradle/wrapper/gradle-wrapper.properties @@ -1,5 +1,5 @@ distributionBase=GRADLE_USER_HOME distributionPath=wrapper/dists -distributionUrl=https\://services.gradle.org/distributions/gradle-8.7-bin.zip +distributionUrl=https\://services.gradle.org/distributions/gradle-8.11-bin.zip zipStoreBase=GRADLE_USER_HOME zipStorePath=wrapper/dists diff --git a/src/main/java/net/F53/HorseBuff/config/ModConfig.java b/src/main/java/net/F53/HorseBuff/config/ModConfig.java index 4016336..2af96b6 100644 --- a/src/main/java/net/F53/HorseBuff/config/ModConfig.java +++ b/src/main/java/net/F53/HorseBuff/config/ModConfig.java @@ -1,85 +1,82 @@ -package net.F53.HorseBuff.config; - -import me.shedaniel.autoconfig.AutoConfig; -import me.shedaniel.autoconfig.serializer.Toml4jConfigSerializer; -import me.shedaniel.autoconfig.ConfigData; -import me.shedaniel.autoconfig.annotation.*; - - -@SuppressWarnings("CanBeFinal") -@Config(name = "HorseBuff") -public class ModConfig implements ConfigData{ - @ConfigEntry.Category("Server") - @ConfigEntry.Gui.Tooltip - public boolean noWander = true; - - @ConfigEntry.Category("Server") - @ConfigEntry.Gui.Tooltip - public boolean portalPatch = true; - - @ConfigEntry.Category("Server") - @ConfigEntry.Gui.Tooltip - public boolean rubberBand = true; - - @ConfigEntry.Category("Server") - @ConfigEntry.Gui.Tooltip - public boolean breakSpeed = true; - - @ConfigEntry.Category("Server") - @ConfigEntry.Gui.Tooltip - public boolean stepHeight = true; - - @ConfigEntry.Category("Client") - @ConfigEntry.Gui.Tooltip - public boolean noBuck = true; - - @ConfigEntry.Category("Client") - @ConfigEntry.Gui.Tooltip - public boolean swimHorse = true; - - @ConfigEntry.Category("Client") - @ConfigEntry.Gui.Tooltip - public boolean swimCamel = false; - - @ConfigEntry.Category("Client") - @ConfigEntry.Gui.Tooltip - public boolean swimDead = false; - - @ConfigEntry.Category("Client") - @ConfigEntry.Gui.Tooltip - @ConfigEntry.Gui.CollapsibleObject - public FadeConfig pitchFade = new FadeConfig(); - - public static class FadeConfig { - public boolean enabled = true; - - @ConfigEntry.Gui.Tooltip - @ConfigEntry.BoundedDiscrete(min = 0, max = 90) - public int startAngle = 30; - - @ConfigEntry.Gui.Tooltip - @ConfigEntry.BoundedDiscrete(min = 0, max = 90) - public int endAngle = 50; - - @ConfigEntry.Gui.Tooltip - @ConfigEntry.BoundedDiscrete(min = 50, max = 100) - public int maxTransparency = 90; - } - - @ConfigEntry.Category("Client") - @ConfigEntry.Gui.Tooltip - @ConfigEntry.BoundedDiscrete(min = 0, max = 45) - public int horseHeadAngleOffset = 0; - - @ConfigEntry.Category("Client") - @ConfigEntry.Gui.Tooltip - public boolean jeb_Horses = true; - - public static void init() { - AutoConfig.register(ModConfig.class, Toml4jConfigSerializer::new); - } - - public static ModConfig getInstance() { - return AutoConfig.getConfigHolder(ModConfig.class).getConfig(); - } -} +package net.F53.HorseBuff.config; + +import me.shedaniel.autoconfig.AutoConfig; +import me.shedaniel.autoconfig.ConfigData; +import me.shedaniel.autoconfig.annotation.Config; +import me.shedaniel.autoconfig.annotation.ConfigEntry; +import me.shedaniel.autoconfig.serializer.Toml4jConfigSerializer; + + +@SuppressWarnings("CanBeFinal") +@Config(name = "HorseBuff") +public class ModConfig implements ConfigData{ + @ConfigEntry.Category("Server") + @ConfigEntry.Gui.Tooltip + public boolean noWander = true; + + @ConfigEntry.Category("Server") + @ConfigEntry.Gui.Tooltip + public boolean rubberBand = true; + + @ConfigEntry.Category("Server") + @ConfigEntry.Gui.Tooltip + public boolean breakSpeed = true; + + @ConfigEntry.Category("Server") + @ConfigEntry.Gui.Tooltip + public boolean stepHeight = true; + + @ConfigEntry.Category("Client") + @ConfigEntry.Gui.Tooltip + public boolean noBuck = true; + + @ConfigEntry.Category("Client") + @ConfigEntry.Gui.Tooltip + public boolean swimHorse = true; + + @ConfigEntry.Category("Client") + @ConfigEntry.Gui.Tooltip + public boolean swimCamel = false; + + @ConfigEntry.Category("Client") + @ConfigEntry.Gui.Tooltip + public boolean swimDead = false; + + @ConfigEntry.Category("Client") + @ConfigEntry.Gui.Tooltip + @ConfigEntry.Gui.CollapsibleObject + public FadeConfig pitchFade = new FadeConfig(); + + public static class FadeConfig { + public boolean enabled = true; + + @ConfigEntry.Gui.Tooltip + @ConfigEntry.BoundedDiscrete(min = 0, max = 90) + public int startAngle = 30; + + @ConfigEntry.Gui.Tooltip + @ConfigEntry.BoundedDiscrete(min = 0, max = 90) + public int endAngle = 50; + + @ConfigEntry.Gui.Tooltip + @ConfigEntry.BoundedDiscrete(min = 50, max = 100) + public int maxTransparency = 90; + } + + @ConfigEntry.Category("Client") + @ConfigEntry.Gui.Tooltip + @ConfigEntry.BoundedDiscrete(min = 0, max = 45) + public int horseHeadAngleOffset = 0; + + @ConfigEntry.Category("Client") + @ConfigEntry.Gui.Tooltip + public boolean jeb_Horses = true; + + public static void init() { + AutoConfig.register(ModConfig.class, Toml4jConfigSerializer::new); + } + + public static ModConfig getInstance() { + return AutoConfig.getConfigHolder(ModConfig.class).getConfig(); + } +} diff --git a/src/main/java/net/F53/HorseBuff/mixin/Client/HeadPitchOffset.java b/src/main/java/net/F53/HorseBuff/mixin/Client/HeadPitchOffset.java index 71951b3..ee75725 100644 --- a/src/main/java/net/F53/HorseBuff/mixin/Client/HeadPitchOffset.java +++ b/src/main/java/net/F53/HorseBuff/mixin/Client/HeadPitchOffset.java @@ -1,12 +1,11 @@ package net.F53.HorseBuff.mixin.Client; import net.F53.HorseBuff.config.ModConfig; +import net.F53.HorseBuff.render.entity.state.ExtendedRideableEntityRenderState; import net.minecraft.client.MinecraftClient; import net.minecraft.client.model.ModelPart; -import net.minecraft.client.render.entity.model.EntityModel; -import net.minecraft.client.render.entity.model.HorseEntityModel; -import net.minecraft.entity.LivingEntity; -import net.minecraft.entity.passive.AbstractHorseEntity; +import net.minecraft.client.render.entity.model.AbstractHorseEntityModel; +import net.minecraft.client.render.entity.state.LivingHorseEntityRenderState; import org.spongepowered.asm.mixin.Final; import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Shadow; @@ -14,15 +13,18 @@ import org.spongepowered.asm.mixin.injection.Inject; import org.spongepowered.asm.mixin.injection.callback.CallbackInfo; +@Mixin(value = AbstractHorseEntityModel.class, priority = 960) +public abstract class HeadPitchOffset { -@Mixin(value = HorseEntityModel.class, priority = 960) -public class HeadPitchOffset> { @Shadow @Final protected ModelPart head; // Thanks dorianpb#9929 for the tip on where to mixin - @Inject(method = "setAngles(Lnet/minecraft/entity/passive/AbstractHorseEntity;FFFFF)V", at = @At("TAIL")) - void headPitch(AbstractHorseEntity horseBaseEntity, float f, float g, float h, float i, float j, CallbackInfo ci){ - if (horseBaseEntity.hasPassenger(MinecraftClient.getInstance().player) && MinecraftClient.getInstance().options.getPerspective().isFirstPerson()) - this.head.pitch = Math.min(this.head.pitch + ModConfig.getInstance().horseHeadAngleOffset/100f, 1.5f); + @Inject(method = "setAngles(Lnet/minecraft/client/render/entity/state/LivingHorseEntityRenderState;)V", at = @At("TAIL")) + void headPitch(LivingHorseEntityRenderState livingHorseEntityRenderState, CallbackInfo ci) { + if (livingHorseEntityRenderState instanceof ExtendedRideableEntityRenderState extendedRideableEntityRenderState) { + if (extendedRideableEntityRenderState.horsebuff$isPlayerPassenger() && MinecraftClient.getInstance().options.getPerspective().isFirstPerson()) { + this.head.pitch = Math.min(this.head.pitch + ModConfig.getInstance().horseHeadAngleOffset / 100f, 1.5f); + } + } } } diff --git a/src/main/java/net/F53/HorseBuff/mixin/Client/HorseArmorFeatureRendererMixin.java b/src/main/java/net/F53/HorseBuff/mixin/Client/HorseArmorFeatureRendererMixin.java new file mode 100644 index 0000000..a39e31b --- /dev/null +++ b/src/main/java/net/F53/HorseBuff/mixin/Client/HorseArmorFeatureRendererMixin.java @@ -0,0 +1,25 @@ +package net.F53.HorseBuff.mixin.Client; + +import com.llamalad7.mixinextras.sugar.Local; +import net.F53.HorseBuff.render.entity.model.ExtendedRideableEquippableEntityModel; +import net.F53.HorseBuff.render.entity.state.ExtendedRideableEntityRenderState; +import net.minecraft.client.render.entity.feature.HorseArmorFeatureRenderer; +import net.minecraft.client.render.entity.model.HorseEntityModel; +import net.minecraft.client.render.entity.state.HorseEntityRenderState; +import org.spongepowered.asm.mixin.Mixin; +import org.spongepowered.asm.mixin.injection.At; +import org.spongepowered.asm.mixin.injection.Inject; +import org.spongepowered.asm.mixin.injection.callback.CallbackInfo; + +@Mixin(HorseArmorFeatureRenderer.class) +public abstract class HorseArmorFeatureRendererMixin { + + @Inject(method = "render(Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumerProvider;ILnet/minecraft/client/render/entity/state/HorseEntityRenderState;FF)V", + at = @At(value = "INVOKE", target = "Lnet/minecraft/client/render/entity/model/HorseEntityModel;setAngles(Lnet/minecraft/client/render/entity/state/LivingHorseEntityRenderState;)V", shift = At.Shift.AFTER)) + void updatePlayerPassenger(CallbackInfo callbackInfo, @Local(argsOnly = true) HorseEntityRenderState horseEntityRenderState, @Local HorseEntityModel horseEntityModel) { + if(horseEntityRenderState instanceof ExtendedRideableEntityRenderState extendedRideableEntityRenderState) { + boolean isPlayerPassenger = extendedRideableEntityRenderState.horsebuff$isPlayerPassenger(); + ((ExtendedRideableEquippableEntityModel) horseEntityModel).horsebuff$setPlayerPassenger(isPlayerPassenger); + } + } +} diff --git a/src/main/java/net/F53/HorseBuff/mixin/Client/HorseRenderer.java b/src/main/java/net/F53/HorseBuff/mixin/Client/HorseRenderer.java deleted file mode 100644 index 10a1275..0000000 --- a/src/main/java/net/F53/HorseBuff/mixin/Client/HorseRenderer.java +++ /dev/null @@ -1,55 +0,0 @@ -package net.F53.HorseBuff.mixin.Client; - -import com.llamalad7.mixinextras.sugar.Local; -import com.llamalad7.mixinextras.sugar.Share; -import com.llamalad7.mixinextras.sugar.ref.LocalIntRef; -import net.F53.HorseBuff.utils.RenderUtils; -import net.minecraft.client.render.entity.LivingEntityRenderer; -import net.minecraft.entity.LivingEntity; -import net.minecraft.entity.passive.AbstractHorseEntity; -import net.minecraft.entity.passive.SheepEntity; -import net.minecraft.util.DyeColor; -import net.minecraft.util.math.ColorHelper.Argb; -import org.spongepowered.asm.mixin.Mixin; -import org.spongepowered.asm.mixin.injection.At; -import org.spongepowered.asm.mixin.injection.Inject; -import org.spongepowered.asm.mixin.injection.ModifyArg; -import org.spongepowered.asm.mixin.injection.callback.CallbackInfo; - -@Mixin(value = LivingEntityRenderer.class, priority = 960) -public abstract class HorseRenderer { - @Inject(method = "render(Lnet/minecraft/entity/LivingEntity;FFLnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumerProvider;I)V", at = @At("HEAD")) - void getAlpha(CallbackInfo ci, @Local(argsOnly = true) LivingEntity entity, @Share("alpha") LocalIntRef alpha) { - if (entity instanceof AbstractHorseEntity) - alpha.set(RenderUtils.getAlpha(entity)); - } - - @ModifyArg(method = "render(Lnet/minecraft/entity/LivingEntity;FFLnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumerProvider;I)V", - at = @At(value = "INVOKE", target = "Lnet/minecraft/client/render/entity/model/EntityModel;render(Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumer;III)V"), - index = 4) - int setOpacityAndChromaForRender(int color, @Local(argsOnly = true, ordinal = 1) float tickDelta, @Local(argsOnly = true) LivingEntity entity, @Share("alpha") LocalIntRef alpha) { - if (!(entity instanceof AbstractHorseEntity)) return color; - if (RenderUtils.isJeb(entity)) { - // see net/minecraft/client/render/entity/feature/SheepWoolFeatureRenderer - int dyeIndex = entity.age / 25 + entity.getId(); - int numDyes = DyeColor.values().length; - int currentDye = SheepEntity.getRgbColor(DyeColor.byId(dyeIndex % numDyes)); - int nextDye = SheepEntity.getRgbColor(DyeColor.byId((dyeIndex + 1) % numDyes)); - float dyeTransitionProgress = ((float) (entity.age % 25) + tickDelta) / 25.0f; - color = Argb.lerp(dyeTransitionProgress, currentDye, nextDye); - // increase brightness by a bit because the horse texture is a bit dark - color = Argb.getArgb(Math.min(Argb.getRed(color) * 2, 255), Math.min(Argb.getGreen(color) * 2, 255), Math.min(Argb.getBlue(color) * 2, 255)); - } - return Argb.withAlpha(alpha.get(), color); - } - - @ModifyArg(method = "render(Lnet/minecraft/entity/LivingEntity;FFLnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumerProvider;I)V", - at = @At(value = "INVOKE", target = "Lnet/minecraft/client/render/entity/LivingEntityRenderer;getRenderLayer(Lnet/minecraft/entity/LivingEntity;ZZZ)Lnet/minecraft/client/render/RenderLayer;"), - index = 2) - boolean makeRenderLayerTranslucent(boolean translucent, @Local(argsOnly = true) LivingEntity entity, @Share("alpha") LocalIntRef alphaRef) { - if (translucent) return true; - if (entity instanceof AbstractHorseEntity) - return alphaRef.get() != 255; - return false; - } -} diff --git a/src/main/java/net/F53/HorseBuff/mixin/Client/InventoryAccessor.java b/src/main/java/net/F53/HorseBuff/mixin/Client/InventoryAccessor.java index 02e5340..adb0489 100644 --- a/src/main/java/net/F53/HorseBuff/mixin/Client/InventoryAccessor.java +++ b/src/main/java/net/F53/HorseBuff/mixin/Client/InventoryAccessor.java @@ -1,5 +1,6 @@ package net.F53.HorseBuff.mixin.Client; +import net.F53.HorseBuff.ClientInit; import net.minecraft.client.MinecraftClient; import net.minecraft.client.gui.screen.Screen; import net.minecraft.client.gui.screen.ingame.InventoryScreen; @@ -11,24 +12,21 @@ import org.spongepowered.asm.mixin.Shadow; import org.spongepowered.asm.mixin.injection.At; import org.spongepowered.asm.mixin.injection.Redirect; -import net.F53.HorseBuff.ClientInit; @Mixin(value = MinecraftClient.class, priority = 960) public abstract class InventoryAccessor { - @Shadow @Final private TutorialManager tutorialManager; + @Shadow @Final private TutorialManager tutorialManager; @Shadow public abstract void setScreen(@Nullable Screen screen); - @Shadow @Nullable public ClientPlayerEntity player; - @Redirect(method= "handleInputEvents()V", at = @At(value = "INVOKE", target = "net/minecraft/client/network/ClientPlayerEntity.openRidingInventory ()V")) - void playerInventoryAccess(ClientPlayerEntity instance){ + @Redirect(method = "handleInputEvents()V", at = @At(value = "INVOKE", target = "net/minecraft/client/network/ClientPlayerEntity.openRidingInventory ()V")) + void playerInventoryAccess(ClientPlayerEntity instance) { assert this.player != null; if (ClientInit.horsePlayerInventory.isPressed()) { tutorialManager.onInventoryOpened(); setScreen(new InventoryScreen(this.player)); - } - else { + } else { instance.openRidingInventory(); } } diff --git a/src/main/java/net/F53/HorseBuff/mixin/Client/JebHorseTintable.java b/src/main/java/net/F53/HorseBuff/mixin/Client/JebHorseTintable.java index 73e375c..08baeaa 100644 --- a/src/main/java/net/F53/HorseBuff/mixin/Client/JebHorseTintable.java +++ b/src/main/java/net/F53/HorseBuff/mixin/Client/JebHorseTintable.java @@ -1,22 +1,30 @@ -package net.F53.HorseBuff.mixin.Client; - -import net.minecraft.client.render.entity.HorseEntityRenderer; -import net.minecraft.entity.passive.HorseColor; -import net.minecraft.entity.passive.HorseEntity; -import org.spongepowered.asm.mixin.Mixin; -import org.spongepowered.asm.mixin.injection.At; -import org.spongepowered.asm.mixin.injection.Redirect; - -import static net.F53.HorseBuff.utils.RenderUtils.isJeb; - -@Mixin(value = HorseEntityRenderer.class, priority = 960) -public class JebHorseTintable { - @Redirect(method = "getTexture(Lnet/minecraft/entity/passive/HorseEntity;)Lnet/minecraft/util/Identifier;", - at = @At(value = "INVOKE", target = "Lnet/minecraft/entity/passive/HorseEntity;getVariant()Lnet/minecraft/entity/passive/HorseColor;")) - HorseColor jebHorseTintable(HorseEntity instance){ - if (isJeb(instance)){ - return HorseColor.WHITE; - } - return instance.getVariant(); - } -} +package net.F53.HorseBuff.mixin.Client; + +import net.minecraft.client.render.entity.HorseEntityRenderer; +import net.minecraft.client.render.entity.state.HorseEntityRenderState; +import net.minecraft.entity.passive.HorseColor; +import net.minecraft.util.Identifier; +import org.spongepowered.asm.mixin.Final; +import org.spongepowered.asm.mixin.Mixin; +import org.spongepowered.asm.mixin.Shadow; +import org.spongepowered.asm.mixin.injection.At; +import org.spongepowered.asm.mixin.injection.Redirect; + +import java.util.Map; + +import static net.F53.HorseBuff.utils.RenderUtils.isJeb; + +@Mixin(value = HorseEntityRenderer.class, priority = 960) +public abstract class JebHorseTintable { + + @Final @Shadow private static Map TEXTURES; + + @Redirect(method = "getTexture(Lnet/minecraft/client/render/entity/state/LivingEntityRenderState;)Lnet/minecraft/util/Identifier;", + at = @At(value = "INVOKE", target = "Lnet/minecraft/client/render/entity/HorseEntityRenderer;getTexture(Lnet/minecraft/client/render/entity/state/HorseEntityRenderState;)Lnet/minecraft/util/Identifier;")) + Identifier jebHorseTintable(HorseEntityRenderer instance, HorseEntityRenderState horseEntityRenderState) { + if (isJeb(horseEntityRenderState)) { + return TEXTURES.get(HorseColor.WHITE); + } + return TEXTURES.get(horseEntityRenderState.color); + } +} diff --git a/src/main/java/net/F53/HorseBuff/mixin/Client/LivingEntityRendererMixin.java b/src/main/java/net/F53/HorseBuff/mixin/Client/LivingEntityRendererMixin.java new file mode 100644 index 0000000..53dad81 --- /dev/null +++ b/src/main/java/net/F53/HorseBuff/mixin/Client/LivingEntityRendererMixin.java @@ -0,0 +1,77 @@ +package net.F53.HorseBuff.mixin.Client; + +import com.llamalad7.mixinextras.sugar.Local; +import com.llamalad7.mixinextras.sugar.Share; +import com.llamalad7.mixinextras.sugar.ref.LocalIntRef; +import net.F53.HorseBuff.render.entity.state.ExtendedRideableEntityRenderState; +import net.F53.HorseBuff.utils.RenderUtils; +import net.minecraft.client.MinecraftClient; +import net.minecraft.client.render.entity.LivingEntityRenderer; +import net.minecraft.client.render.entity.state.LivingEntityRenderState; +import net.minecraft.entity.LivingEntity; +import net.minecraft.entity.passive.SheepEntity; +import net.minecraft.util.DyeColor; +import net.minecraft.util.math.ColorHelper; +import net.minecraft.util.math.MathHelper; +import org.spongepowered.asm.mixin.Mixin; +import org.spongepowered.asm.mixin.injection.At; +import org.spongepowered.asm.mixin.injection.Inject; +import org.spongepowered.asm.mixin.injection.ModifyArg; +import org.spongepowered.asm.mixin.injection.callback.CallbackInfo; + +import static net.F53.HorseBuff.utils.RenderUtils.isJeb; +import static net.F53.HorseBuff.utils.RenderUtils.isRideableEntityRenderState; + +@Mixin(value = LivingEntityRenderer.class, priority = 960) +public abstract class LivingEntityRendererMixin { + + @Inject(method = "updateRenderState(Lnet/minecraft/entity/LivingEntity;Lnet/minecraft/client/render/entity/state/LivingEntityRenderState;F)V", + at = @At("TAIL")) + void setPlayerPassenger(CallbackInfo ci, @Local(argsOnly = true) LivingEntityRenderState livingEntityRenderState, @Local(argsOnly = true) LivingEntity livingEntity) { + if (isRideableEntityRenderState(livingEntityRenderState)) { + boolean playerPassenger = livingEntity.hasPassenger(MinecraftClient.getInstance().player); + ((ExtendedRideableEntityRenderState) livingEntityRenderState).horsebuff$setId(livingEntity.getId()); + ((ExtendedRideableEntityRenderState) livingEntityRenderState).horsebuff$setPlayerPassenger(playerPassenger); + } + } + + @Inject(method = "render(Lnet/minecraft/client/render/entity/state/LivingEntityRenderState;Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumerProvider;I)V", + at = @At("HEAD")) + void setAlpha(CallbackInfo ci, @Local(argsOnly = true) LivingEntityRenderState livingEntityRenderState, @Share("alpha") LocalIntRef alpha) { + if (livingEntityRenderState instanceof ExtendedRideableEntityRenderState extendedRideableEntityRenderState) { + alpha.set(RenderUtils.getAlpha(extendedRideableEntityRenderState.horsebuff$isPlayerPassenger())); + } + } + + @ModifyArg(method = "render(Lnet/minecraft/client/render/entity/state/LivingEntityRenderState;Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumerProvider;I)V", + at = @At(value = "INVOKE", target = "Lnet/minecraft/client/render/entity/model/EntityModel;render(Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumer;III)V"), + index = 4) + int setOpacityAndChromaForRender(int color, @Local(argsOnly = true) LivingEntityRenderState livingEntityRenderState, @Share("alpha") LocalIntRef alpha) { + if (isRideableEntityRenderState(livingEntityRenderState)) { + if (isJeb(livingEntityRenderState) && livingEntityRenderState instanceof ExtendedRideableEntityRenderState extendedRideableEntityRenderState) { + // see net/minecraft/client/render/entity/feature/SheepWoolFeatureRenderer + int dyeIndex = MathHelper.floor(livingEntityRenderState.age) / 25 + extendedRideableEntityRenderState.horsebuff$getId(); + int numDyes = DyeColor.values().length; + int currentDye = SheepEntity.getRgbColor(DyeColor.byId(dyeIndex % numDyes)); + int nextDye = SheepEntity.getRgbColor(DyeColor.byId((dyeIndex + 1) % numDyes)); + float dyeTransitionProgress = ((float) (MathHelper.floor(livingEntityRenderState.age) % 25) + MathHelper.fractionalPart(livingEntityRenderState.age)) / 25.0F; + color = ColorHelper.lerp(dyeTransitionProgress, currentDye, nextDye); + // increase brightness by a bit because the horse texture is a bit dark + color = ColorHelper.getArgb(Math.min(ColorHelper.getRed(color) * 2, 255), Math.min(ColorHelper.getGreen(color) * 2, 255), Math.min(ColorHelper.getBlue(color) * 2, 255)); + } + return ColorHelper.withAlpha(alpha.get(), color); + } else { + return color; + } + } + + @ModifyArg(method = "render(Lnet/minecraft/client/render/entity/state/LivingEntityRenderState;Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumerProvider;I)V", + at = @At(value = "INVOKE", target = "Lnet/minecraft/client/render/entity/LivingEntityRenderer;getRenderLayer(Lnet/minecraft/client/render/entity/state/LivingEntityRenderState;ZZZ)Lnet/minecraft/client/render/RenderLayer;"), + index = 2) + boolean makeRenderLayerTranslucent(boolean translucent, @Local(argsOnly = true) LivingEntityRenderState livingEntityRenderState, @Share("alpha") LocalIntRef alphaRef) { + if (translucent) return true; + if (isRideableEntityRenderState(livingEntityRenderState)) + return alphaRef.get() != 255; + return false; + } +} diff --git a/src/main/java/net/F53/HorseBuff/mixin/Client/LlamaDecorFeatureRendererMixin.java b/src/main/java/net/F53/HorseBuff/mixin/Client/LlamaDecorFeatureRendererMixin.java new file mode 100644 index 0000000..5fc4d5d --- /dev/null +++ b/src/main/java/net/F53/HorseBuff/mixin/Client/LlamaDecorFeatureRendererMixin.java @@ -0,0 +1,26 @@ +package net.F53.HorseBuff.mixin.Client; + +import com.llamalad7.mixinextras.sugar.Local; +import net.F53.HorseBuff.render.entity.model.ExtendedRideableEquippableEntityModel; +import net.F53.HorseBuff.render.entity.state.ExtendedRideableEntityRenderState; +import net.minecraft.client.render.entity.feature.LlamaDecorFeatureRenderer; +import net.minecraft.client.render.entity.model.LlamaEntityModel; +import net.minecraft.client.render.entity.state.LlamaEntityRenderState; +import org.spongepowered.asm.mixin.Mixin; +import org.spongepowered.asm.mixin.injection.At; +import org.spongepowered.asm.mixin.injection.Inject; +import org.spongepowered.asm.mixin.injection.callback.CallbackInfo; + +@Mixin(value = LlamaDecorFeatureRenderer.class, priority = 960) +public class LlamaDecorFeatureRendererMixin { + + @Inject(method = "render(Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumerProvider;Lnet/minecraft/client/render/entity/state/LlamaEntityRenderState;Lnet/minecraft/item/ItemStack;Lnet/minecraft/util/Identifier;I)V", + at = @At(value = "INVOKE", target = "Lnet/minecraft/client/render/entity/model/LlamaEntityModel;setAngles(Lnet/minecraft/client/render/entity/state/LlamaEntityRenderState;)V", + shift = At.Shift.AFTER)) + void updatePlayerPassenger(CallbackInfo callbackInfo, @Local(argsOnly = true) LlamaEntityRenderState llamaEntityRenderState, @Local LlamaEntityModel llamaEntityModel) { + if (llamaEntityRenderState instanceof ExtendedRideableEntityRenderState extendedRideableEntityRenderState) { + boolean isPlayerPassenger = extendedRideableEntityRenderState.horsebuff$isPlayerPassenger(); + ((ExtendedRideableEquippableEntityModel) llamaEntityModel).horsebuff$setPlayerPassenger(isPlayerPassenger); + } + } +} diff --git a/src/main/java/net/F53/HorseBuff/mixin/Client/RideableEntityRenderStateMixin.java b/src/main/java/net/F53/HorseBuff/mixin/Client/RideableEntityRenderStateMixin.java new file mode 100644 index 0000000..33ff29c --- /dev/null +++ b/src/main/java/net/F53/HorseBuff/mixin/Client/RideableEntityRenderStateMixin.java @@ -0,0 +1,41 @@ +package net.F53.HorseBuff.mixin.Client; + +import net.F53.HorseBuff.render.entity.state.ExtendedRideableEntityRenderState; +import net.minecraft.client.render.entity.state.CamelEntityRenderState; +import net.minecraft.client.render.entity.state.LivingHorseEntityRenderState; +import net.minecraft.client.render.entity.state.LlamaEntityRenderState; +import org.spongepowered.asm.mixin.Mixin; +import org.spongepowered.asm.mixin.Unique; + +// Additional fields for: +// horses, donkeys, mules, undead horses, skeleton horses (LivingHorseEntityRenderState) +// llamas and camels +@Mixin(value = {LivingHorseEntityRenderState.class, LlamaEntityRenderState.class, CamelEntityRenderState.class}, priority = 960) +public abstract class RideableEntityRenderStateMixin implements ExtendedRideableEntityRenderState { + + @Unique + private int horsebuff$id; + + @Unique + private boolean horsebuff$playerPassenger; + + @Override + public void horsebuff$setId(int id) { + this.horsebuff$id = id; + } + + @Override + public int horsebuff$getId() { + return this.horsebuff$id; + } + + @Override + public void horsebuff$setPlayerPassenger(boolean playerPassenger) { + this.horsebuff$playerPassenger = playerPassenger; + } + + @Override + public boolean horsebuff$isPlayerPassenger() { + return horsebuff$playerPassenger; + } +} diff --git a/src/main/java/net/F53/HorseBuff/mixin/Client/RideableEquippableEntityModelMixin.java b/src/main/java/net/F53/HorseBuff/mixin/Client/RideableEquippableEntityModelMixin.java new file mode 100644 index 0000000..8af509b --- /dev/null +++ b/src/main/java/net/F53/HorseBuff/mixin/Client/RideableEquippableEntityModelMixin.java @@ -0,0 +1,25 @@ +package net.F53.HorseBuff.mixin.Client; + +import net.F53.HorseBuff.render.entity.model.ExtendedRideableEquippableEntityModel; +import net.minecraft.client.render.entity.model.HorseEntityModel; +import net.minecraft.client.render.entity.model.LlamaEntityModel; +import org.spongepowered.asm.mixin.Mixin; +import org.spongepowered.asm.mixin.Unique; + +// Horses can equip armor and llamas can equip carpets +@Mixin(value = {HorseEntityModel.class, LlamaEntityModel.class}, priority = 960) +public abstract class RideableEquippableEntityModelMixin implements ExtendedRideableEquippableEntityModel { + + @Unique + private boolean horsebuff$playerPassenger; + + @Override + public void horsebuff$setPlayerPassenger(boolean playerPassenger) { + this.horsebuff$playerPassenger = playerPassenger; + } + + @Override + public boolean horsebuff$isPlayerPassenger() { + return this.horsebuff$playerPassenger; + } +} diff --git a/src/main/java/net/F53/HorseBuff/mixin/Client/Swim.java b/src/main/java/net/F53/HorseBuff/mixin/Client/Swim.java index 37319bd..a00b235 100644 --- a/src/main/java/net/F53/HorseBuff/mixin/Client/Swim.java +++ b/src/main/java/net/F53/HorseBuff/mixin/Client/Swim.java @@ -4,7 +4,11 @@ import net.minecraft.entity.LivingEntity; import net.minecraft.entity.mob.SkeletonHorseEntity; import net.minecraft.entity.mob.ZombieHorseEntity; -import net.minecraft.entity.passive.*; +import net.minecraft.entity.passive.AbstractHorseEntity; +import net.minecraft.entity.passive.CamelEntity; +import net.minecraft.entity.passive.DonkeyEntity; +import net.minecraft.entity.passive.HorseEntity; +import net.minecraft.entity.passive.MuleEntity; import net.minecraft.entity.player.PlayerEntity; import net.minecraft.registry.tag.FluidTags; import net.minecraft.util.math.Vec3d; @@ -15,12 +19,13 @@ import org.spongepowered.asm.mixin.injection.callback.CallbackInfo; @Mixin(value = LivingEntity.class) -public class Swim { +public abstract class Swim { + @Inject(method = "travelControlled", at = @At(value = "INVOKE", target = "Lnet/minecraft/entity/LivingEntity;travel(Lnet/minecraft/util/math/Vec3d;)V", shift = At.Shift.BEFORE)) private void fakeSwim(PlayerEntity controllingPlayer, Vec3d movementInput, CallbackInfo ci) { if (!((Object)this instanceof AbstractHorseEntity)) {return;} AbstractHorseEntity horseInstance = (AbstractHorseEntity) (Object) this; - if (!hb$shouldSwim(horseInstance)) {return;} + if (!horsebuff$shouldSwim(horseInstance)) {return;} if (horseInstance.getFluidHeight(FluidTags.WATER) > horseInstance.getSwimHeight()) { horseInstance.addVelocity(0, 0.08, 0); @@ -28,7 +33,7 @@ private void fakeSwim(PlayerEntity controllingPlayer, Vec3d movementInput, Callb } @Unique - private boolean hb$shouldSwim(AbstractHorseEntity horseInstance) { + private boolean horsebuff$shouldSwim(AbstractHorseEntity horseInstance) { if (horseInstance instanceof HorseEntity || horseInstance instanceof DonkeyEntity || horseInstance instanceof MuleEntity) { diff --git a/src/main/java/net/F53/HorseBuff/mixin/Client/TransparentArmor.java b/src/main/java/net/F53/HorseBuff/mixin/Client/TransparentArmor.java deleted file mode 100644 index ea0fbf8..0000000 --- a/src/main/java/net/F53/HorseBuff/mixin/Client/TransparentArmor.java +++ /dev/null @@ -1,35 +0,0 @@ -package net.F53.HorseBuff.mixin.Client; - -import com.llamalad7.mixinextras.injector.wrapoperation.Operation; -import com.llamalad7.mixinextras.injector.wrapoperation.WrapOperation; -import com.llamalad7.mixinextras.sugar.Local; -import com.llamalad7.mixinextras.sugar.Share; -import com.llamalad7.mixinextras.sugar.ref.LocalIntRef; -import net.minecraft.client.render.RenderLayer; -import net.minecraft.client.render.entity.feature.HorseArmorFeatureRenderer; -import net.minecraft.entity.passive.HorseEntity; -import net.minecraft.util.Identifier; -import net.minecraft.util.math.ColorHelper.Argb; -import org.spongepowered.asm.mixin.Mixin; -import org.spongepowered.asm.mixin.injection.At; -import org.spongepowered.asm.mixin.injection.ModifyArg; - -import static net.F53.HorseBuff.utils.RenderUtils.getAlpha; - -@Mixin(value = HorseArmorFeatureRenderer.class, priority = 960) -public class TransparentArmor { - @WrapOperation(method = "render(Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumerProvider;ILnet/minecraft/entity/passive/HorseEntity;FFFFFF)V", - at = @At(value = "INVOKE", target = "net/minecraft/client/render/RenderLayer.getEntityCutoutNoCull (Lnet/minecraft/util/Identifier;)Lnet/minecraft/client/render/RenderLayer;")) - RenderLayer makeRenderLayerTranslucent(Identifier texture, Operation original, @Local(argsOnly = true) HorseEntity horseEntity, @Share("alpha") LocalIntRef alpha) { - alpha.set(getAlpha(horseEntity)); - if (alpha.get() == 255) return original.call(texture); - return RenderLayer.getItemEntityTranslucentCull(texture); - } - - @ModifyArg(method = "render(Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumerProvider;ILnet/minecraft/entity/passive/HorseEntity;FFFFFF)V", - at = @At(value = "INVOKE", target = "Lnet/minecraft/client/render/entity/model/HorseEntityModel;render(Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumer;III)V"), - index = 4) - int setOpacityForRender(int color, @Share("alpha") LocalIntRef alpha) { - return Argb.withAlpha(Math.min(Math.max(0, Argb.getAlpha(color)), alpha.get()), color); - } -} diff --git a/src/main/java/net/F53/HorseBuff/mixin/Client/TransparentEquipment.java b/src/main/java/net/F53/HorseBuff/mixin/Client/TransparentEquipment.java new file mode 100644 index 0000000..da14ac4 --- /dev/null +++ b/src/main/java/net/F53/HorseBuff/mixin/Client/TransparentEquipment.java @@ -0,0 +1,43 @@ +package net.F53.HorseBuff.mixin.Client; + +import com.llamalad7.mixinextras.injector.wrapoperation.Operation; +import com.llamalad7.mixinextras.injector.wrapoperation.WrapOperation; +import com.llamalad7.mixinextras.sugar.Local; +import com.llamalad7.mixinextras.sugar.Share; +import com.llamalad7.mixinextras.sugar.ref.LocalIntRef; +import net.F53.HorseBuff.render.entity.model.ExtendedRideableEquippableEntityModel; +import net.F53.HorseBuff.utils.RenderUtils; +import net.minecraft.client.model.Model; +import net.minecraft.client.render.RenderLayer; +import net.minecraft.client.render.entity.equipment.EquipmentRenderer; +import net.minecraft.client.render.entity.model.HorseEntityModel; +import net.minecraft.client.render.entity.model.LlamaEntityModel; +import net.minecraft.util.Identifier; +import net.minecraft.util.math.ColorHelper; +import org.spongepowered.asm.mixin.Mixin; +import org.spongepowered.asm.mixin.injection.At; +import org.spongepowered.asm.mixin.injection.ModifyArg; + +@Mixin(value = EquipmentRenderer.class, priority = 960) +public abstract class TransparentEquipment { + + @WrapOperation(method = "render(Lnet/minecraft/item/equipment/EquipmentModel$LayerType;Lnet/minecraft/util/Identifier;Lnet/minecraft/client/model/Model;Lnet/minecraft/item/ItemStack;Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumerProvider;ILnet/minecraft/util/Identifier;)V", + at = @At(value = "INVOKE", target = "Lnet/minecraft/client/render/RenderLayer;getArmorCutoutNoCull(Lnet/minecraft/util/Identifier;)Lnet/minecraft/client/render/RenderLayer;")) + RenderLayer makeRenderLayerTranslucent(Identifier texture, Operation original, @Local(argsOnly = true) Model model, @Share("alpha") LocalIntRef alpha) { + if(model instanceof ExtendedRideableEquippableEntityModel extendedRideableEquippableEntityModel) { + alpha.set(RenderUtils.getAlpha(extendedRideableEquippableEntityModel.horsebuff$isPlayerPassenger())); + } + if (alpha.get() == 255) return original.call(texture); + return RenderLayer.getItemEntityTranslucentCull(texture); + } + + @ModifyArg(method = "render(Lnet/minecraft/item/equipment/EquipmentModel$LayerType;Lnet/minecraft/util/Identifier;Lnet/minecraft/client/model/Model;Lnet/minecraft/item/ItemStack;Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumerProvider;ILnet/minecraft/util/Identifier;)V", + at = @At(value = "INVOKE", target = "Lnet/minecraft/client/model/Model;render(Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumer;III)V"), + index = 4) + int setOpacityForRender(int color, @Local(argsOnly = true) Model model, @Share("alpha") LocalIntRef alpha) { + if(model instanceof HorseEntityModel || model instanceof LlamaEntityModel) { + return ColorHelper.withAlpha(Math.min(Math.max(0, ColorHelper.getAlpha(color)), alpha.get()), color); + } + return color; + } +} diff --git a/src/main/java/net/F53/HorseBuff/mixin/Client/TransparentMarkings.java b/src/main/java/net/F53/HorseBuff/mixin/Client/TransparentHorseMarkings.java similarity index 68% rename from src/main/java/net/F53/HorseBuff/mixin/Client/TransparentMarkings.java rename to src/main/java/net/F53/HorseBuff/mixin/Client/TransparentHorseMarkings.java index cffd412..0db0e40 100644 --- a/src/main/java/net/F53/HorseBuff/mixin/Client/TransparentMarkings.java +++ b/src/main/java/net/F53/HorseBuff/mixin/Client/TransparentHorseMarkings.java @@ -5,32 +5,34 @@ import com.llamalad7.mixinextras.sugar.Local; import com.llamalad7.mixinextras.sugar.Share; import com.llamalad7.mixinextras.sugar.ref.LocalIntRef; +import net.F53.HorseBuff.render.entity.state.ExtendedRideableEntityRenderState; import net.minecraft.client.render.RenderLayer; import net.minecraft.client.render.VertexConsumer; import net.minecraft.client.render.entity.feature.HorseMarkingFeatureRenderer; import net.minecraft.client.render.entity.model.HorseEntityModel; +import net.minecraft.client.render.entity.state.HorseEntityRenderState; import net.minecraft.client.util.math.MatrixStack; -import net.minecraft.entity.passive.HorseEntity; import net.minecraft.util.Identifier; -import net.minecraft.util.math.ColorHelper.Argb; +import net.minecraft.util.math.ColorHelper; import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.injection.At; import static net.F53.HorseBuff.utils.RenderUtils.getAlpha; @Mixin(value = HorseMarkingFeatureRenderer.class, priority = 960) -public class TransparentMarkings { - @WrapOperation(method = "render(Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumerProvider;ILnet/minecraft/entity/passive/HorseEntity;FFFFFF)V", +public abstract class TransparentHorseMarkings { + + @WrapOperation(method = "render(Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumerProvider;ILnet/minecraft/client/render/entity/state/HorseEntityRenderState;FF)V", at = @At(value = "INVOKE", target = "Lnet/minecraft/client/render/RenderLayer;getEntityTranslucent(Lnet/minecraft/util/Identifier;)Lnet/minecraft/client/render/RenderLayer;")) - RenderLayer makeRenderLayerTranslucent(Identifier texture, Operation original, @Local(argsOnly = true) HorseEntity horseEntity, @Share("alpha") LocalIntRef alpha) { - alpha.set(getAlpha(horseEntity)); + RenderLayer makeRenderLayerTranslucent(Identifier texture, Operation original, @Local(argsOnly = true) HorseEntityRenderState horseEntityRenderState, @Share("alpha") LocalIntRef alpha) { + alpha.set(getAlpha(((ExtendedRideableEntityRenderState) horseEntityRenderState).horsebuff$isPlayerPassenger())); if (alpha.get() == 255) return original.call(texture); return RenderLayer.getItemEntityTranslucentCull(texture); } - @WrapOperation(method = "render(Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumerProvider;ILnet/minecraft/entity/passive/HorseEntity;FFFFFF)V", + @WrapOperation(method = "render(Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumerProvider;ILnet/minecraft/client/render/entity/state/HorseEntityRenderState;FF)V", at = @At(value = "INVOKE", target = "Lnet/minecraft/client/render/entity/model/HorseEntityModel;render(Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumer;II)V")) - void modifyOverlayAlpha(HorseEntityModel instance, MatrixStack matrixStack, VertexConsumer vertexConsumer, int light, int overlay, Operation original, @Share("alpha") LocalIntRef alpha) { - instance.render(matrixStack, vertexConsumer, light, overlay, Argb.withAlpha(alpha.get(), 0xFFFFFF)); + void modifyOverlayAlpha(HorseEntityModel instance, MatrixStack matrixStack, VertexConsumer vertexConsumer, int light, int overlay, Operation original, @Share("alpha") LocalIntRef alpha) { + instance.render(matrixStack, vertexConsumer, light, overlay, ColorHelper.withAlpha(alpha.get(), 0xFFFFFF)); } } diff --git a/src/main/java/net/F53/HorseBuff/mixin/Client/TransparentLlamaDecor.java b/src/main/java/net/F53/HorseBuff/mixin/Client/TransparentLlamaDecor.java deleted file mode 100644 index 93a9485..0000000 --- a/src/main/java/net/F53/HorseBuff/mixin/Client/TransparentLlamaDecor.java +++ /dev/null @@ -1,36 +0,0 @@ -package net.F53.HorseBuff.mixin.Client; - -import com.llamalad7.mixinextras.injector.wrapoperation.Operation; -import com.llamalad7.mixinextras.injector.wrapoperation.WrapOperation; -import com.llamalad7.mixinextras.sugar.Local; -import com.llamalad7.mixinextras.sugar.Share; -import com.llamalad7.mixinextras.sugar.ref.LocalIntRef; -import net.minecraft.client.render.RenderLayer; -import net.minecraft.client.render.VertexConsumer; -import net.minecraft.client.render.entity.feature.LlamaDecorFeatureRenderer; -import net.minecraft.client.render.entity.model.LlamaEntityModel; -import net.minecraft.client.util.math.MatrixStack; -import net.minecraft.entity.passive.LlamaEntity; -import net.minecraft.util.Identifier; -import net.minecraft.util.math.ColorHelper; -import org.spongepowered.asm.mixin.Mixin; -import org.spongepowered.asm.mixin.injection.At; - -import static net.F53.HorseBuff.utils.RenderUtils.getAlpha; - -@Mixin(value = LlamaDecorFeatureRenderer.class, priority = 960) -public class TransparentLlamaDecor { - @WrapOperation(method = "render(Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumerProvider;ILnet/minecraft/entity/passive/LlamaEntity;FFFFFF)V", - at = @At(value = "INVOKE", target = "Lnet/minecraft/client/render/RenderLayer;getEntityCutoutNoCull(Lnet/minecraft/util/Identifier;)Lnet/minecraft/client/render/RenderLayer;")) - RenderLayer modifyOverlayLayer(Identifier texture, Operation original, @Local(argsOnly = true) LlamaEntity llamaEntity, @Share("alpha") LocalIntRef alpha) { - alpha.set(getAlpha(llamaEntity)); - if (alpha.get() == 255) return original.call(texture); - return RenderLayer.getItemEntityTranslucentCull(texture); - } - - @WrapOperation(method = "render(Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumerProvider;ILnet/minecraft/entity/passive/LlamaEntity;FFFFFF)V", - at = @At(value = "INVOKE", target = "Lnet/minecraft/client/render/entity/model/LlamaEntityModel;render(Lnet/minecraft/client/util/math/MatrixStack;Lnet/minecraft/client/render/VertexConsumer;II)V")) - void modifyOverlayAlpha(LlamaEntityModel instance, MatrixStack matrixStack, VertexConsumer vertexConsumer, int light, int overlay, Operation original, @Share("alpha") LocalIntRef alpha) { - instance.render(matrixStack, vertexConsumer, light, overlay, ColorHelper.Argb.withAlpha(alpha.get(), 0xFFFFFF)); - } -} diff --git a/src/main/java/net/F53/HorseBuff/mixin/Server/AllowRaft.java b/src/main/java/net/F53/HorseBuff/mixin/Server/AllowRaft.java index fce8a02..f772d5a 100644 --- a/src/main/java/net/F53/HorseBuff/mixin/Server/AllowRaft.java +++ b/src/main/java/net/F53/HorseBuff/mixin/Server/AllowRaft.java @@ -4,22 +4,24 @@ import com.llamalad7.mixinextras.sugar.Local; import net.minecraft.entity.Entity; import net.minecraft.entity.passive.AbstractHorseEntity; -import net.minecraft.entity.vehicle.BoatEntity; +import net.minecraft.entity.vehicle.AbstractBoatEntity; +import net.minecraft.entity.vehicle.RaftEntity; import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Unique; import org.spongepowered.asm.mixin.injection.At; -@Mixin(BoatEntity.class) +@Mixin(AbstractBoatEntity.class) public abstract class AllowRaft { + @ModifyReturnValue(method = "isSmallerThanBoat", at = @At("RETURN")) - private boolean allowRaft(boolean original, @Local(argsOnly = true)Entity entity){ - if (entity instanceof AbstractHorseEntity && hb$thiz().getVariant() == BoatEntity.Type.BAMBOO) + private boolean allowRaft(boolean original, @Local(argsOnly = true) Entity entity) { + if (entity instanceof AbstractHorseEntity && horsebuff$thiz() instanceof RaftEntity) return true; return original; } @Unique - private BoatEntity hb$thiz() { - return ((BoatEntity)(Object)this); + private AbstractBoatEntity horsebuff$thiz() { + return ((AbstractBoatEntity) (Object) this); } } diff --git a/src/main/java/net/F53/HorseBuff/mixin/Server/MountedModifiers.java b/src/main/java/net/F53/HorseBuff/mixin/Server/MountedModifiers.java index b128e0a..084edae 100644 --- a/src/main/java/net/F53/HorseBuff/mixin/Server/MountedModifiers.java +++ b/src/main/java/net/F53/HorseBuff/mixin/Server/MountedModifiers.java @@ -20,14 +20,15 @@ // - remove BreakSpeed debuff from not being grounded (when enabled) @Mixin(value = PlayerEntity.class, priority = 960) public abstract class MountedModifiers extends LivingEntity { + protected MountedModifiers(EntityType entityType, World world) { super(entityType, world); } @Unique - EntityAttributeModifier mountedStepHeight = new EntityAttributeModifier(new Identifier("horse-buff", "mounted-step-height"), 0.1, EntityAttributeModifier.Operation.ADD_VALUE); + EntityAttributeModifier horsebuff$mountedStepHeight = new EntityAttributeModifier(Identifier.of("horse-buff", "mounted-step-height"), 0.1, EntityAttributeModifier.Operation.ADD_VALUE); @Unique - EntityAttributeModifier mountedBreakSpeed = new EntityAttributeModifier(new Identifier("horse-buff", "mounted-break-speed"), 5, EntityAttributeModifier.Operation.ADD_MULTIPLIED_BASE); + EntityAttributeModifier horsebuff$mountedBreakSpeed = new EntityAttributeModifier(Identifier.of("horse-buff", "mounted-break-speed"), 5, EntityAttributeModifier.Operation.ADD_MULTIPLIED_BASE); @Override public boolean startRiding(Entity entity, boolean force) { @@ -36,13 +37,13 @@ public boolean startRiding(Entity entity, boolean force) { return result; if (ModConfig.getInstance().stepHeight) { - EntityAttributeInstance stepHeight = horse.getAttributeInstance(EntityAttributes.GENERIC_STEP_HEIGHT); - if (stepHeight != null) stepHeight.addTemporaryModifier(mountedStepHeight); + EntityAttributeInstance stepHeight = horse.getAttributeInstance(EntityAttributes.STEP_HEIGHT); + if (stepHeight != null) stepHeight.addTemporaryModifier(horsebuff$mountedStepHeight); } if (ModConfig.getInstance().breakSpeed) { - EntityAttributeInstance breakSpeed = getAttributeInstance(EntityAttributes.PLAYER_BLOCK_BREAK_SPEED); - if (breakSpeed != null) breakSpeed.addTemporaryModifier(mountedBreakSpeed); + EntityAttributeInstance breakSpeed = getAttributeInstance(EntityAttributes.BLOCK_BREAK_SPEED); + if (breakSpeed != null) breakSpeed.addTemporaryModifier(horsebuff$mountedBreakSpeed); } return result; } @@ -59,11 +60,11 @@ public void stopRiding() { return; } - EntityAttributeInstance stepHeight = horse.getAttributeInstance(EntityAttributes.GENERIC_STEP_HEIGHT); - if (stepHeight != null) stepHeight.removeModifier(mountedStepHeight); + EntityAttributeInstance stepHeight = horse.getAttributeInstance(EntityAttributes.STEP_HEIGHT); + if (stepHeight != null) stepHeight.removeModifier(horsebuff$mountedStepHeight); - EntityAttributeInstance breakSpeed = getAttributeInstance(EntityAttributes.PLAYER_BLOCK_BREAK_SPEED); - if (breakSpeed != null) breakSpeed.removeModifier(mountedBreakSpeed); + EntityAttributeInstance breakSpeed = getAttributeInstance(EntityAttributes.BLOCK_BREAK_SPEED); + if (breakSpeed != null) breakSpeed.removeModifier(horsebuff$mountedBreakSpeed); super.stopRiding(); } diff --git a/src/main/java/net/F53/HorseBuff/mixin/Server/MovementCheck.java b/src/main/java/net/F53/HorseBuff/mixin/Server/MovementCheck.java index 391be71..8b34fd0 100644 --- a/src/main/java/net/F53/HorseBuff/mixin/Server/MovementCheck.java +++ b/src/main/java/net/F53/HorseBuff/mixin/Server/MovementCheck.java @@ -11,7 +11,8 @@ // Disable movement checks for Players on Horses, fixing MC-100830 @Mixin(value = ServerPlayNetworkHandler.class, priority = 960) -public class MovementCheck { +public abstract class MovementCheck { + @Shadow public ServerPlayerEntity player; // TODO: figure out safer alternative to ModifyConstant here (not easy) diff --git a/src/main/java/net/F53/HorseBuff/mixin/Server/NoBuck.java b/src/main/java/net/F53/HorseBuff/mixin/Server/NoBuck.java index 9eeca1f..de57972 100644 --- a/src/main/java/net/F53/HorseBuff/mixin/Server/NoBuck.java +++ b/src/main/java/net/F53/HorseBuff/mixin/Server/NoBuck.java @@ -1,29 +1,28 @@ -package net.F53.HorseBuff.mixin.Server; - -import com.llamalad7.mixinextras.injector.ModifyReturnValue; -import net.F53.HorseBuff.config.ModConfig; -import net.minecraft.entity.LivingEntity; -import net.minecraft.entity.passive.AbstractHorseEntity; -import org.spongepowered.asm.mixin.Mixin; -import org.spongepowered.asm.mixin.Unique; -import org.spongepowered.asm.mixin.injection.At; - -@Mixin(value = AbstractHorseEntity.class, priority = 960) -public class NoBuck { - @ModifyReturnValue(method = "isAngry", at = @At("RETURN")) - private boolean isAngry(boolean original) { - AbstractHorseEntity instance = hb$thiz(); - if (ModConfig.getInstance().noBuck - && !instance.jumping - && instance.isTame() - && instance.getControllingPassenger() != null) { - return false; - } - return original; - } - - @Unique - private AbstractHorseEntity hb$thiz() { - return ((AbstractHorseEntity)(Object)this); - } -} +package net.F53.HorseBuff.mixin.Server; + +import com.llamalad7.mixinextras.injector.ModifyReturnValue; +import net.F53.HorseBuff.config.ModConfig; +import net.minecraft.entity.LivingEntity; +import net.minecraft.entity.passive.AbstractHorseEntity; +import org.spongepowered.asm.mixin.Mixin; +import org.spongepowered.asm.mixin.Shadow; +import org.spongepowered.asm.mixin.injection.At; + +@Mixin(value = AbstractHorseEntity.class, priority = 960) +public abstract class NoBuck { + + @Shadow protected boolean jumping; + @Shadow public abstract boolean isTame(); + @Shadow public abstract LivingEntity getControllingPassenger(); + + @ModifyReturnValue(method = "isAngry", at = @At("RETURN")) + private boolean isAngry(boolean original) { + if (ModConfig.getInstance().noBuck + && jumping + && isTame() + && getControllingPassenger() != null) { + return false; + } + return original; + } +} diff --git a/src/main/java/net/F53/HorseBuff/mixin/Server/NoWander.java b/src/main/java/net/F53/HorseBuff/mixin/Server/NoWander.java index baad7dd..7c13fcc 100644 --- a/src/main/java/net/F53/HorseBuff/mixin/Server/NoWander.java +++ b/src/main/java/net/F53/HorseBuff/mixin/Server/NoWander.java @@ -12,17 +12,18 @@ // Lower wander speed for saddled horses @Mixin(value = LivingEntity.class, priority = 960) public abstract class NoWander { + @ModifyArg(method = "tickMovement", at = @At(value = "INVOKE", target = "Lnet/minecraft/entity/LivingEntity;travel(Lnet/minecraft/util/math/Vec3d;)V")) private Vec3d lowerWanderSpeed(Vec3d input) { if (ModConfig.getInstance().noWander - && (hb$thiz() instanceof AbstractHorseEntity horse + && (horsebuff$thiz() instanceof AbstractHorseEntity horse && horse.isSaddled())) return(Vec3d.ZERO); return input; } @Unique - private LivingEntity hb$thiz() { + private LivingEntity horsebuff$thiz() { return ((LivingEntity)(Object)this); } } diff --git a/src/main/java/net/F53/HorseBuff/render/entity/model/ExtendedRideableEquippableEntityModel.java b/src/main/java/net/F53/HorseBuff/render/entity/model/ExtendedRideableEquippableEntityModel.java new file mode 100644 index 0000000..8bfac30 --- /dev/null +++ b/src/main/java/net/F53/HorseBuff/render/entity/model/ExtendedRideableEquippableEntityModel.java @@ -0,0 +1,6 @@ +package net.F53.HorseBuff.render.entity.model; + +public interface ExtendedRideableEquippableEntityModel { + void horsebuff$setPlayerPassenger(boolean playerPassenger); + boolean horsebuff$isPlayerPassenger(); +} diff --git a/src/main/java/net/F53/HorseBuff/render/entity/state/ExtendedRideableEntityRenderState.java b/src/main/java/net/F53/HorseBuff/render/entity/state/ExtendedRideableEntityRenderState.java new file mode 100644 index 0000000..e3a6dd0 --- /dev/null +++ b/src/main/java/net/F53/HorseBuff/render/entity/state/ExtendedRideableEntityRenderState.java @@ -0,0 +1,8 @@ +package net.F53.HorseBuff.render.entity.state; + +public interface ExtendedRideableEntityRenderState { + void horsebuff$setId(int id); + int horsebuff$getId(); + void horsebuff$setPlayerPassenger(boolean playerPassenger); + boolean horsebuff$isPlayerPassenger(); +} diff --git a/src/main/java/net/F53/HorseBuff/utils/RenderUtils.java b/src/main/java/net/F53/HorseBuff/utils/RenderUtils.java index 88e5d33..64b9304 100644 --- a/src/main/java/net/F53/HorseBuff/utils/RenderUtils.java +++ b/src/main/java/net/F53/HorseBuff/utils/RenderUtils.java @@ -1,25 +1,34 @@ -package net.F53.HorseBuff.utils; - -import net.F53.HorseBuff.config.ModConfig; -import net.minecraft.client.MinecraftClient; -import net.minecraft.entity.Entity; - -public class RenderUtils { - public static boolean isJeb(Entity entity) { - return ModConfig.getInstance().jeb_Horses && entity.hasCustomName() && "jeb_".equals(entity.getName().getString()); - } - - public static int getAlpha(Entity horse) { - ModConfig.FadeConfig pitchFadeConfig = ModConfig.getInstance().pitchFade; - if (!pitchFadeConfig.enabled) return 255; - MinecraftClient client = MinecraftClient.getInstance(); - if (client.player == null || !client.options.getPerspective().isFirstPerson() || !horse.hasPassenger(client.player)) - return 255; - - int minAlpha = 255 - Math.round(pitchFadeConfig.maxTransparency * 2.25f); - int rate = (255 - minAlpha) / (pitchFadeConfig.startAngle - pitchFadeConfig.endAngle); - int unclampedAlpha = Math.round(rate * (client.player.renderPitch - pitchFadeConfig.endAngle)); - - return Math.min(Math.max(unclampedAlpha, minAlpha), 255); - } -} +package net.F53.HorseBuff.utils; + +import net.F53.HorseBuff.config.ModConfig; +import net.minecraft.client.MinecraftClient; +import net.minecraft.client.render.entity.state.CamelEntityRenderState; +import net.minecraft.client.render.entity.state.LivingEntityRenderState; +import net.minecraft.client.render.entity.state.LivingHorseEntityRenderState; +import net.minecraft.client.render.entity.state.LlamaEntityRenderState; + +public class RenderUtils { + public static boolean isJeb(LivingEntityRenderState entityRenderState) { + return ModConfig.getInstance().jeb_Horses && entityRenderState.customName != null && "jeb_".equals(entityRenderState.customName.getString()); + } + + public static int getAlpha(boolean isPlayerPassenger) { + ModConfig.FadeConfig pitchFadeConfig = ModConfig.getInstance().pitchFade; + if (!pitchFadeConfig.enabled) return 255; + MinecraftClient client = MinecraftClient.getInstance(); + if (client.player == null || !client.options.getPerspective().isFirstPerson() || !isPlayerPassenger) + return 255; + + int minAlpha = 255 - Math.round(pitchFadeConfig.maxTransparency * 2.25f); + int rate = (255 - minAlpha) / (pitchFadeConfig.startAngle - pitchFadeConfig.endAngle); + int unclampedAlpha = Math.round(rate * (client.player.renderPitch - pitchFadeConfig.endAngle)); + + return Math.min(Math.max(unclampedAlpha, minAlpha), 255); + } + + public static boolean isRideableEntityRenderState(LivingEntityRenderState livingEntityRenderState) { + return livingEntityRenderState instanceof LivingHorseEntityRenderState || + livingEntityRenderState instanceof LlamaEntityRenderState || + livingEntityRenderState instanceof CamelEntityRenderState; + } +} diff --git a/src/main/resources/fabric.mod.json b/src/main/resources/fabric.mod.json index 627f1d2..978efc5 100644 --- a/src/main/resources/fabric.mod.json +++ b/src/main/resources/fabric.mod.json @@ -2,7 +2,6 @@ "schemaVersion": 1, "id": "horsebuff", "version": "${version}", - "accessWidener": "horsebuff.accesswidener", "name": "HorseBuff", "description": "Makes Horses better by removing stupid things", "authors": [ @@ -31,8 +30,8 @@ ], "depends": { "fabricloader": ">=0.15.11", - "minecraft": ">=1.21", - "java": ">=17", + "minecraft": ">=1.21.3", + "java": ">=21", "fabric-api": "*", "cloth-config": "*" }, diff --git a/src/main/resources/horsebuff.accesswidener b/src/main/resources/horsebuff.accesswidener deleted file mode 100644 index 229ad34..0000000 --- a/src/main/resources/horsebuff.accesswidener +++ /dev/null @@ -1,5 +0,0 @@ -accessWidener v2 named - -accessible method net/minecraft/util/Identifier (Ljava/lang/String;Ljava/lang/String;)V - -accessible field net/minecraft/entity/passive/AbstractHorseEntity jumping Z diff --git a/src/main/resources/horsebuff.mixins.json b/src/main/resources/horsebuff.mixins.json index 4b606ef..72c9628 100644 --- a/src/main/resources/horsebuff.mixins.json +++ b/src/main/resources/horsebuff.mixins.json @@ -2,7 +2,7 @@ "required": true, "minVersion": "0.8", "package": "net.F53.HorseBuff.mixin", - "compatibilityLevel": "JAVA_16", + "compatibilityLevel": "JAVA_21", "mixins": [ "Server.AllowRaft", "Server.MountedModifiers", @@ -12,13 +12,16 @@ ], "client": [ "Client.HeadPitchOffset", - "Client.HorseRenderer", + "Client.HorseArmorFeatureRendererMixin", "Client.InventoryAccessor", "Client.JebHorseTintable", + "Client.LivingEntityRendererMixin", + "Client.LlamaDecorFeatureRendererMixin", + "Client.RideableEquippableEntityModelMixin", + "Client.RideableEntityRenderStateMixin", "Client.Swim", - "Client.TransparentArmor", - "Client.TransparentLlamaDecor", - "Client.TransparentMarkings" + "Client.TransparentEquipment", + "Client.TransparentHorseMarkings" ], "injectors": { "defaultRequire": 1