diff --git a/src/main/java/shcm/shsupercm/fabric/citresewn/ActiveCITs.java b/src/main/java/shcm/shsupercm/fabric/citresewn/ActiveCITs.java index 63627645..8d56a356 100644 --- a/src/main/java/shcm/shsupercm/fabric/citresewn/ActiveCITs.java +++ b/src/main/java/shcm/shsupercm/fabric/citresewn/ActiveCITs.java @@ -179,8 +179,8 @@ public void setEnchantmentAppliedContextCached(ItemStack stack, World world, Liv return; } - for (CITEnchantment cit : citEnchantments) - cit.applyMethod(stack); + if (effectiveGlobalProperties.method != null) + effectiveGlobalProperties.method.applyMethod(citEnchantments, stack); CITEnchantment.appliedContext = citEnchantments; } diff --git a/src/main/java/shcm/shsupercm/fabric/citresewn/mixin/citenchantment/MinecraftClientMixin.java b/src/main/java/shcm/shsupercm/fabric/citresewn/mixin/citenchantment/MinecraftClientMixin.java new file mode 100644 index 00000000..a9bea56c --- /dev/null +++ b/src/main/java/shcm/shsupercm/fabric/citresewn/mixin/citenchantment/MinecraftClientMixin.java @@ -0,0 +1,17 @@ +package shcm.shsupercm.fabric.citresewn.mixin.citenchantment; + +import net.minecraft.client.MinecraftClient; +import org.spongepowered.asm.mixin.Mixin; +import org.spongepowered.asm.mixin.injection.At; +import org.spongepowered.asm.mixin.injection.Inject; +import org.spongepowered.asm.mixin.injection.callback.CallbackInfo; + +import static shcm.shsupercm.fabric.citresewn.pack.cits.CITEnchantment.MergeMethod.ticks; + +@Mixin(MinecraftClient.class) +public class MinecraftClientMixin { + @Inject(method = "tick", at = @At("HEAD")) + public void onTick(CallbackInfo ci) { + ticks++; + } +} diff --git a/src/main/java/shcm/shsupercm/fabric/citresewn/pack/cits/CITEnchantment.java b/src/main/java/shcm/shsupercm/fabric/citresewn/pack/cits/CITEnchantment.java index 759d6cc0..7c4fa947 100644 --- a/src/main/java/shcm/shsupercm/fabric/citresewn/pack/cits/CITEnchantment.java +++ b/src/main/java/shcm/shsupercm/fabric/citresewn/pack/cits/CITEnchantment.java @@ -15,7 +15,6 @@ import net.minecraft.util.math.Matrix4f; import net.minecraft.util.math.Vec3f; import org.lwjgl.opengl.GL11; -import shcm.shsupercm.fabric.citresewn.CITResewn; import shcm.shsupercm.fabric.citresewn.config.CITResewnConfig; import shcm.shsupercm.fabric.citresewn.ex.CITParseException; import shcm.shsupercm.fabric.citresewn.mixin.citenchantment.BufferBuilderStorageAccessor; @@ -28,6 +27,7 @@ import static org.lwjgl.opengl.GL11.*; import static com.mojang.blaze3d.systems.RenderSystem.*; +import static shcm.shsupercm.util.logic.Loops.statelessFadingLoop; public class CITEnchantment extends CIT { public static List appliedContext = null; @@ -40,7 +40,6 @@ public class CITEnchantment extends CIT { public final Blend blend; private final WrappedMethodIntensity methodIntensity = new WrappedMethodIntensity(); - private final MergeMethod method; public final Map renderLayers = new EnumMap<>(GlintRenderLayer.class); @@ -77,8 +76,6 @@ public CITEnchantment(CITPack pack, Identifier identifier, Properties properties }; blend = Blend.getBlend(properties.getProperty("blend", "add")); - - method = !enchantmentsAny && this.enchantments.size() > 0 ? pack.method : null; } catch (Exception e) { throw new CITParseException(pack.resourcePack, identifier, (e.getClass() == Exception.class ? "" : e.getClass().getSimpleName() + ": ") + e.getMessage()); } @@ -93,17 +90,6 @@ public void activate() { } } - public void applyMethod(ItemStack stack) { - if (this.method != null) { - Map stackEnchantments = new LinkedHashMap<>(); - for (NbtElement nbtElement : stack.isOf(Items.ENCHANTED_BOOK) ? EnchantedBookItem.getEnchantmentNbt(stack) : stack.getEnchantments()) - stackEnchantments.put(EnchantmentHelper.getIdFromNbt((NbtCompound) nbtElement), EnchantmentHelper.getLevelFromNbt((NbtCompound) nbtElement)); - - this.methodIntensity.intensity = this.method.getIntensity(stackEnchantments, this); - } else - this.methodIntensity.intensity = 1f; - } - @Override public void dispose() { appliedContext = null; @@ -195,10 +181,10 @@ public RenderLayer build(CITEnchantment enchantment) { } public VertexConsumer tryApply(VertexConsumer base, RenderLayer baseLayer, VertexConsumerProvider provider) { - if (!shouldApply || appliedContext == null) + if (!shouldApply || appliedContext == null || appliedContext.size() == 0) return null; - VertexConsumer[] layers = new VertexConsumer[Math.min(appliedContext.size(), CITResewn.INSTANCE.activeCITs.effectiveGlobalProperties.cap)]; + VertexConsumer[] layers = new VertexConsumer[Math.min(appliedContext.size(), appliedContext.get(0).pack.cap)]; for (int i = 0; i < layers.length; i++) layers[i] = provider.getBuffer(appliedContext.get(i).renderLayers.get(GlintRenderLayer.this)); @@ -305,51 +291,74 @@ public BlendFormatException() { public enum MergeMethod { AVERAGE { @Override - public float getIntensity(Map stackEnchantments, CITEnchantment cit) { + public void applyIntensity(Map stackEnchantments, CITEnchantment cit) { Identifier enchantment = null; for (Identifier enchantmentMatch : cit.enchantments) if (stackEnchantments.containsKey(enchantmentMatch)) { enchantment = enchantmentMatch; break; } - if (enchantment == null) - return 0f; - float sum = 0f; - for (Integer value : stackEnchantments.values()) - sum += value; + if (enchantment == null) { + cit.methodIntensity.intensity = 0f; + } else { + float sum = 0f; + for (Integer value : stackEnchantments.values()) + sum += value; - return (float) stackEnchantments.get(enchantment) / sum; + cit.methodIntensity.intensity = (float) stackEnchantments.get(enchantment) / sum; + } } }, LAYERED { @Override - public float getIntensity(Map stackEnchantments, CITEnchantment cit) { + public void applyIntensity(Map stackEnchantments, CITEnchantment cit) { Identifier enchantment = null; for (Identifier enchantmentMatch : cit.enchantments) if (stackEnchantments.containsKey(enchantmentMatch)) { enchantment = enchantmentMatch; break; } - if (enchantment == null) - return 0f; + if (enchantment == null) { + cit.methodIntensity.intensity = 0f; + return; + } float max = 0f; for (Integer value : stackEnchantments.values()) if (value > max) max = value; - return (float) stackEnchantments.get(enchantment) / max; + cit.methodIntensity.intensity = (float) stackEnchantments.get(enchantment) / max; } }, CYCLE { @Override - public float getIntensity(Map stackEnchantments, CITEnchantment cit) { - return 1f; + public void applyMethod(List citEnchantments, ItemStack stack) { + List> durations = new ArrayList<>(); + for (CITEnchantment cit : citEnchantments) + durations.add(new HashMap.SimpleEntry<>(cit, cit.duration)); + + for (Map.Entry intensity : statelessFadingLoop(durations, citEnchantments.get(0).pack.fade, ticks, 20).entrySet()) + intensity.getKey().methodIntensity.intensity = intensity.getValue(); } }; - public abstract float getIntensity(Map stackEnchantments, CITEnchantment cit); + public static int ticks = 0; + + public void applyIntensity(Map stackEnchantments, CITEnchantment cit) { + cit.methodIntensity.intensity = 1f; + } + + public void applyMethod(List citEnchantments, ItemStack stack) { + Map stackEnchantments = new LinkedHashMap<>(); + for (NbtElement nbtElement : stack.isOf(Items.ENCHANTED_BOOK) ? EnchantedBookItem.getEnchantmentNbt(stack) : stack.getEnchantments()) + stackEnchantments.put(EnchantmentHelper.getIdFromNbt((NbtCompound) nbtElement), EnchantmentHelper.getLevelFromNbt((NbtCompound) nbtElement)); + + for (CITEnchantment cit : citEnchantments) + if (!cit.enchantmentsAny) + applyIntensity(stackEnchantments, cit); + } } private static class WrappedMethodIntensity { diff --git a/src/main/java/shcm/shsupercm/util/logic/Loops.java b/src/main/java/shcm/shsupercm/util/logic/Loops.java new file mode 100644 index 00000000..6c182441 --- /dev/null +++ b/src/main/java/shcm/shsupercm/util/logic/Loops.java @@ -0,0 +1,60 @@ +package shcm.shsupercm.util.logic; + +import java.util.*; + +/** + * This class(or class portion) is a part of SHCM's utilities. Feel free to use without credit. + */ +public class Loops { + /** + * Creates a loop of T with linked intensities allowing for fading between the elements. + * @param items list of items and pause durations(in time units) ordered as they are in the loop + * @param fade time in units to fade between each item + * @param ticks positive raising counter + * @param tpu the amount of ticks per time unit + * @param element type + * @return map of elements and their respective intensities(between 0.0f and 1.0f) + */ + public static Map statelessFadingLoop(List> items, float fade, int ticks, int tpu) { + Map itemValues = new HashMap<>(); + + if (items == null || items.size() == 0) + return itemValues; + + if (items.size() == 1) { + itemValues.put(items.get(0).getKey(), 1f); + return itemValues; + } + + float totalUnitsInLoop = 0f; + for (Map.Entry item : items) { + itemValues.put(item.getKey(), 0f); + totalUnitsInLoop += item.getValue() + fade; + } + + float unitInLoop = (ticks % (tpu * totalUnitsInLoop)) / tpu; + + for (int i = 0; i < items.size(); i++) { + Map.Entry item = items.get(i); + if (unitInLoop < item.getValue()) { + itemValues.put(item.getKey(), 1f); + break; + } else + unitInLoop -= item.getValue(); + + if (unitInLoop < fade) { + Map.Entry nextItem = items.get(i + 1 >= items.size() ? 0 : i + 1); + + unitInLoop /= fade; + + itemValues.put(item.getKey(), 1f - unitInLoop); + itemValues.put(nextItem.getKey(), unitInLoop); + + break; + } else + unitInLoop -= fade; + } + + return itemValues; + } +} \ No newline at end of file diff --git a/src/main/resources/citresewn.mixins.json b/src/main/resources/citresewn.mixins.json index 179a3bf0..e32c8471 100644 --- a/src/main/resources/citresewn.mixins.json +++ b/src/main/resources/citresewn.mixins.json @@ -19,6 +19,7 @@ "citenchantment.ItemRendererMixin", "citenchantment.ItemStackMixin", "citenchantment.RenderPhaseAccessor", + "citenchantment.MinecraftClientMixin", "cititem.ItemRendererMixin", "cititem.ItemStackMixin", "cititem.JsonUnbakedModelAccessor",