Skip to content

Commit

Permalink
更新 molang 和动画系统
Browse files Browse the repository at this point in the history
- 现在新的动画系统支持更加完备的基岩版动画
  • Loading branch information
TartaricAcid committed Oct 27, 2024
1 parent e6f37e6 commit 91352bb
Show file tree
Hide file tree
Showing 225 changed files with 7,797 additions and 3,184 deletions.

Large diffs are not rendered by default.

Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
package com.github.tartaricacid.touhoulittlemaid.client.animation.gecko.molang;

import com.github.tartaricacid.touhoulittlemaid.geckolib3.core.molang.binding.ContextBinding;

public class TLMBinding extends ContextBinding {
public static final TLMBinding INSTANCE = new TLMBinding();

private TLMBinding() {
maidEntityVar("is_begging", ctx -> ctx.entity().isBegging());
maidEntityVar("is_sitting", ctx -> ctx.entity().isMaidInSittingPose());
maidEntityVar("has_backpack", ctx -> ctx.entity().hasBackpack());
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,131 @@
package com.github.tartaricacid.touhoulittlemaid.client.animation.gecko.molang;

import com.github.tartaricacid.touhoulittlemaid.client.animation.gecko.molang.functions.*;
import com.github.tartaricacid.touhoulittlemaid.client.animation.gecko.molang.variable.LadderFacingVariable;
import com.github.tartaricacid.touhoulittlemaid.geckolib3.core.event.predicate.AnimationEvent;
import com.github.tartaricacid.touhoulittlemaid.geckolib3.core.molang.binding.ContextBinding;
import com.github.tartaricacid.touhoulittlemaid.geckolib3.core.molang.builtin.query.EmptyFunction;
import com.github.tartaricacid.touhoulittlemaid.util.EquipmentUtil;
import net.minecraft.client.Minecraft;
import net.minecraft.client.multiplayer.ClientLevel;
import net.minecraft.core.BlockPos;
import net.minecraft.world.entity.Entity;
import net.minecraft.world.entity.EquipmentSlot;
import net.minecraft.world.entity.LivingEntity;
import net.minecraft.world.entity.Pose;
import net.minecraft.world.entity.ai.attributes.Attributes;
import net.minecraft.world.level.levelgen.Heightmap;
import net.neoforged.neoforge.common.NeoForgeMod;
import org.apache.commons.lang3.StringUtils;

public class YSMBinding extends ContextBinding {
public static final YSMBinding INSTANCE = new YSMBinding();

@SuppressWarnings("resource")
private YSMBinding() {
function("mod_version", new ModVersion());
function("equipped_enchantment_level", new EquippedEnchantmentLevel());
function("effect_level", new EffectLevel());
function("relative_block_name", new RelativeBlockName());

function("bone_rot", new BoneRotation());
function("bone_pos", new BonePosition());
function("bone_scale", new BoneScale());

var("head_yaw", ctx -> ctx.data().netHeadYaw);
var("head_pitch", ctx -> ctx.data().headPitch);
var("weather", ctx -> getWeather(ctx.level()));
var("dimension_name", ctx -> ctx.level().dimension().location().toString());
var("fps", ctx -> Minecraft.getInstance().getFps());

entityVar("is_passenger", ctx -> ctx.entity().isPassenger());
entityVar("is_sleep", ctx -> ctx.entity().getPose() == Pose.SLEEPING);
entityVar("is_sneak", ctx -> ctx.entity().onGround() && ctx.entity().getPose() == Pose.CROUCHING);
entityVar("is_open_air", ctx -> isOpenAir(ctx.entity()));
entityVar("eye_in_water", ctx -> ctx.entity().isUnderWater());
entityVar("frozen_ticks", ctx -> ctx.entity().getTicksFrozen());
entityVar("air_supply", ctx -> ctx.entity().getAirSupply());

livingEntityVar("has_helmet", ctx -> getSlotValue(ctx.entity(), EquipmentSlot.HEAD));
livingEntityVar("has_chest_plate", ctx -> getSlotValue(ctx.entity(), EquipmentSlot.CHEST));
livingEntityVar("has_leggings", ctx -> getSlotValue(ctx.entity(), EquipmentSlot.LEGS));
livingEntityVar("has_boots", ctx -> getSlotValue(ctx.entity(), EquipmentSlot.FEET));
livingEntityVar("has_mainhand", ctx -> getSlotValue(ctx.entity(), EquipmentSlot.MAINHAND));
livingEntityVar("has_offhand", ctx -> getSlotValue(ctx.entity(), EquipmentSlot.OFFHAND));
livingEntityVar("has_elytra", ctx -> !EquipmentUtil.getEquippedElytraItem(ctx.entity()).isEmpty());
livingEntityVar("is_riptide", ctx -> ctx.entity().isAutoSpinAttack());
livingEntityVar("armor_value", ctx -> ctx.entity().getArmorValue());
livingEntityVar("hurt_time", ctx -> ctx.entity().hurtTime);
livingEntityVar("is_close_eyes", ctx -> getEyeCloseState(ctx.animationEvent(), ctx.entity()));
livingEntityVar("on_ladder", ctx -> ctx.entity().onClimbable());
livingEntityVar("ladder_facing", new LadderFacingVariable());
livingEntityVar("arrow_count", ctx -> ctx.entity().getArrowCount());
livingEntityVar("stinger_count", ctx -> ctx.entity().getStingerCount());

livingEntityVar("attack_damage", ctx -> ctx.entity().getAttributeValue(Attributes.ATTACK_DAMAGE));
livingEntityVar("attack_speed", ctx -> ctx.entity().getAttributeValue(Attributes.ATTACK_SPEED));
livingEntityVar("attack_knockback", ctx -> ctx.entity().getAttributeValue(Attributes.ATTACK_KNOCKBACK));
livingEntityVar("movement_speed", ctx -> ctx.entity().getAttributeValue(Attributes.MOVEMENT_SPEED));
livingEntityVar("knockback_resistance", ctx -> ctx.entity().getAttributeValue(Attributes.KNOCKBACK_RESISTANCE));
livingEntityVar("luck", ctx -> ctx.entity().getAttributeValue(Attributes.LUCK));

livingEntityVar("block_reach", ctx -> ctx.entity().getAttributeValue(Attributes.BLOCK_INTERACTION_RANGE));
livingEntityVar("entity_reach", ctx -> ctx.entity().getAttributeValue(Attributes.ENTITY_INTERACTION_RANGE));
livingEntityVar("swim_speed", ctx -> ctx.entity().getAttributeValue(NeoForgeMod.SWIM_SPEED));
livingEntityVar("entity_gravity", ctx -> ctx.entity().getAttributeValue(Attributes.GRAVITY));
livingEntityVar("step_height_addition", ctx -> ctx.entity().getAttributeValue(Attributes.STEP_HEIGHT) - 0.6);
livingEntityVar("nametag_distance", ctx -> ctx.entity().getAttributeValue(NeoForgeMod.NAMETAG_DISTANCE));


// 女仆和 YSM 之间不一致的 molang,仅保留防止报错
function("dump_equipped_item", new EmptyFunction());
function("dump_relative_block", new EmptyFunction());
function("bone_pivot_abs", new EmptyFunction());

var("dump_mods", ctx -> 0);
var("texture_name", ctx -> StringUtils.EMPTY);
var("elytra_rot_x", ctx -> 0);
var("elytra_rot_y", ctx -> 0);
var("elytra_rot_z", ctx -> 0);

entityVar("dump_effects", ctx -> 0);
entityVar("dump_biome", ctx -> 0);
entityVar("biome_category", ctx -> 0);

livingEntityVar("rendering_in_inventory", ctx -> false);
maidEntityVar("food_level", ctx -> 20);

var("first_person_mod_hide", ctx -> false);
var("has_left_shoulder_parrot", ctx -> false);
var("has_right_shoulder_parrot", ctx -> false);
var("left_shoulder_parrot_variant", ctx -> 0);
var("right_shoulder_parrot_variant", ctx -> 0);
}

private static boolean getEyeCloseState(AnimationEvent<?> animationEvent, LivingEntity player) {
double remainder = (animationEvent.getAnimationTick() + Math.abs(player.getUUID().getLeastSignificantBits()) % 10) % 90;
boolean isBlinkTime = 85 < remainder && remainder < 90;
return player.isSleeping() || isBlinkTime;
}

private static boolean getSlotValue(LivingEntity entity, EquipmentSlot slot) {
return !EquipmentUtil.getEquippedItem(entity, slot).isEmpty();
}

private static int getWeather(ClientLevel world) {
if (world.isThundering()) {
return 2;
} else if (world.isRaining()) {
return 1;
}
return 0;
}

private static boolean isOpenAir(Entity entity) {
BlockPos blockpos = entity.blockPosition();
if (!entity.level.canSeeSky(blockpos)) {
return false;
}
return entity.level.getHeightmapPos(Heightmap.Types.MOTION_BLOCKING, blockpos).getY() <= blockpos.getY();
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
package com.github.tartaricacid.touhoulittlemaid.client.animation.gecko.molang.functions;

import com.github.tartaricacid.touhoulittlemaid.client.animation.gecko.molang.struct.Vec3fStruct;
import com.github.tartaricacid.touhoulittlemaid.geckolib3.core.molang.context.IContext;
import com.github.tartaricacid.touhoulittlemaid.geckolib3.core.molang.function.entity.EntityFunction;
import com.github.tartaricacid.touhoulittlemaid.geckolib3.core.processor.IBone;
import com.github.tartaricacid.touhoulittlemaid.molang.runtime.ExecutionContext;
import net.minecraft.util.StringUtil;
import net.minecraft.world.entity.Entity;
import org.jetbrains.annotations.NotNull;

public abstract class BoneParamFunction extends EntityFunction {
@Override
public boolean validateArgumentSize(int size) {
return size == 1;
}

@Override
protected Object eval(ExecutionContext<IContext<Entity>> context, ArgumentCollection arguments) {
var str = arguments.getAsString(context, 0);
if (StringUtil.isNullOrEmpty(str)) {
return null;
}

var bone = context.entity().geoInstance().getAnimationProcessor().getBone(str);
if (bone == null) {
return null;
}

return getParam(bone);
}

protected abstract Vec3fStruct getParam(@NotNull IBone bone);
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
package com.github.tartaricacid.touhoulittlemaid.client.animation.gecko.molang.functions;

import com.github.tartaricacid.touhoulittlemaid.client.animation.gecko.molang.struct.Vec3fStruct;
import com.github.tartaricacid.touhoulittlemaid.geckolib3.core.processor.IBone;
import org.jetbrains.annotations.NotNull;

public final class BonePosition extends BoneParamFunction {
@Override
protected Vec3fStruct getParam(@NotNull IBone bone) {
return new BonePositionStruct(bone);
}

private static final class BonePositionStruct extends Vec3fStruct {
private final IBone bone;

public BonePositionStruct(IBone bone) {
this.bone = bone;
}

@Override
protected float getX() {
return bone.getPositionX();
}

@Override
protected float getY() {
return bone.getPositionY();
}

@Override
protected float getZ() {
return bone.getPositionZ();
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
package com.github.tartaricacid.touhoulittlemaid.client.animation.gecko.molang.functions;

import com.github.tartaricacid.touhoulittlemaid.client.animation.gecko.molang.struct.Vec3fStruct;
import com.github.tartaricacid.touhoulittlemaid.geckolib3.core.processor.IBone;
import org.jetbrains.annotations.NotNull;

public final class BoneRotation extends BoneParamFunction {
@Override
protected Vec3fStruct getParam(@NotNull IBone bone) {
return new BoneRotationStruct(bone);
}

private static final class BoneRotationStruct extends Vec3fStruct {
private final IBone bone;

public BoneRotationStruct(IBone bone) {
this.bone = bone;
}

@Override
protected float getX() {
return -(float) Math.toDegrees(bone.getRotationX());
}

@Override
protected float getY() {
return -(float) Math.toDegrees(bone.getRotationY());
}

@Override
protected float getZ() {
return (float) Math.toDegrees(bone.getRotationZ());
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
package com.github.tartaricacid.touhoulittlemaid.client.animation.gecko.molang.functions;

import com.github.tartaricacid.touhoulittlemaid.client.animation.gecko.molang.struct.Vec3fStruct;
import com.github.tartaricacid.touhoulittlemaid.geckolib3.core.processor.IBone;
import org.jetbrains.annotations.NotNull;

public final class BoneScale extends BoneParamFunction {
@Override
protected Vec3fStruct getParam(@NotNull IBone bone) {
return new BoneScaleStruct(bone);
}

private static final class BoneScaleStruct extends Vec3fStruct {
private final IBone bone;

public BoneScaleStruct(IBone bone) {
this.bone = bone;
}

@Override
protected float getX() {
return bone.getScaleX();
}

@Override
protected float getY() {
return bone.getScaleY();
}

@Override
protected float getZ() {
return bone.getScaleZ();
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
package com.github.tartaricacid.touhoulittlemaid.client.animation.gecko.molang.functions;

import com.github.tartaricacid.touhoulittlemaid.geckolib3.core.molang.context.IContext;
import com.github.tartaricacid.touhoulittlemaid.geckolib3.core.molang.function.ContextFunction;
import com.github.tartaricacid.touhoulittlemaid.geckolib3.util.MolangUtils;
import com.github.tartaricacid.touhoulittlemaid.molang.runtime.ExecutionContext;
import net.minecraft.core.Holder;
import net.minecraft.core.registries.BuiltInRegistries;
import net.minecraft.resources.ResourceLocation;
import net.minecraft.world.effect.MobEffect;
import net.minecraft.world.effect.MobEffectInstance;
import net.minecraft.world.entity.Entity;
import net.minecraft.world.entity.LivingEntity;

public class EffectLevel extends ContextFunction<Entity> {
@Override
public boolean validateArgumentSize(int size) {
return size == 1;
}

@Override
protected Object eval(ExecutionContext<IContext<Entity>> context, ArgumentCollection arguments) {
ResourceLocation effectId = MolangUtils.parseResourceLocation(context.entity(), arguments.getAsString(context, 0));
if (effectId == null) {
return null;
}

Holder.Reference<MobEffect> effect = BuiltInRegistries.MOB_EFFECT.getHolder(effectId).orElse(null);
if (effect == null) {
return 0;
}

if (context.entity().entity() instanceof LivingEntity) {
MobEffectInstance instance = ((LivingEntity) context.entity().entity()).getEffect(effect);
if (instance != null) {
return instance.getAmplifier() + 1;
}
} else {
return null;
}

return 0;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
package com.github.tartaricacid.touhoulittlemaid.client.animation.gecko.molang.functions;

import com.github.tartaricacid.touhoulittlemaid.geckolib3.core.molang.context.IContext;
import com.github.tartaricacid.touhoulittlemaid.geckolib3.core.molang.function.entity.LivingEntityFunction;
import com.github.tartaricacid.touhoulittlemaid.geckolib3.util.MolangUtils;
import com.github.tartaricacid.touhoulittlemaid.molang.runtime.ExecutionContext;
import com.github.tartaricacid.touhoulittlemaid.util.EquipmentUtil;
import net.minecraft.core.Holder;
import net.minecraft.core.HolderLookup;
import net.minecraft.core.registries.Registries;
import net.minecraft.resources.ResourceKey;
import net.minecraft.resources.ResourceLocation;
import net.minecraft.world.entity.EquipmentSlot;
import net.minecraft.world.entity.LivingEntity;
import net.minecraft.world.item.ItemStack;
import net.minecraft.world.item.enchantment.Enchantment;

public class EquippedEnchantmentLevel extends LivingEntityFunction {
@Override
protected Object eval(ExecutionContext<IContext<LivingEntity>> context, ArgumentCollection arguments) {
EquipmentSlot slotType = MolangUtils.parseSlotType(context.entity(), arguments.getAsString(context, 0));
if (slotType == null) {
return null;
}

ResourceLocation id = MolangUtils.parseResourceLocation(context.entity(), arguments.getAsString(context, 1));
if (id == null) {
return null;
}
HolderLookup.RegistryLookup<Enchantment> lookup = context.entity().level().registryAccess().lookup(Registries.ENCHANTMENT).orElse(null);
if (lookup == null) {
return 0;
}

ResourceKey<Enchantment> resourceKey = ResourceKey.create(Registries.ENCHANTMENT, id);
Holder.Reference<Enchantment> enchantment = lookup.get(resourceKey).orElse(null);
if (enchantment == null) {
return 0;
}

ItemStack itemStack = EquipmentUtil.getEquippedItem(context.entity().entity(), slotType);
if (itemStack.isEmpty()) {
return 0;
}

return itemStack.getEnchantmentLevel(enchantment);
}

@Override
public boolean validateArgumentSize(int size) {
return size == 2;
}
}
Loading

0 comments on commit 91352bb

Please sign in to comment.