using HarmonyLib; using System; using System.Collections.Generic; using System.Linq; using System.Reflection; using System.Reflection.Emit; using System.Text; using System.Threading.Tasks; using UnityEngine; using Verse; using AlienRace; namespace Rimworld_Animations { [HarmonyPatch(typeof(AlienRace.HarmonyPatches), "DrawAddons")] public static class HarmonyPatch_AlienRace { public static void RenderHeadAddonInAnimation(Mesh mesh, Vector3 loc, Quaternion quat, Material mat, bool drawNow, Graphic graphic, AlienPartGenerator.BodyAddon bodyAddon, Vector3 v, float num, Vector3 headOffset, Pawn pawn, PawnRenderFlags renderFlags) { CompBodyAnimator pawnAnimator = pawn.TryGetComp(); if (pawnAnimator != null && !renderFlags.FlagSet(PawnRenderFlags.Portrait) && pawnAnimator.isAnimating && (bodyAddon.drawnInBed || bodyAddon.alignWithHead)) { Quaternion headQuatInAnimation = Quaternion.AngleAxis(pawnAnimator.headAngle, Vector3.up); Rot4 headRotInAnimation = pawnAnimator.headFacing; Vector3 headPositionInAnimation = pawnAnimator.getPawnHeadPosition(); headPositionInAnimation.y = loc.y; Vector3 orassanv = Vector3.zero; bool orassan = false; if ((pawn.def as ThingDef_AlienRace).defName == "Alien_Orassan") { orassan = true; if(bodyAddon.path.Contains("closed")) { return; } if (bodyAddon.bodyPart.Contains("ear")) { orassan = true; orassanv = new Vector3(0, 0, 0.23f); if (pawnAnimator.headFacing == Rot4.North) { orassanv.z -= 0.1f; orassanv.y += 1f; if(bodyAddon.bodyPart.Contains("left")) { orassanv.x += 0.03f; } else { orassanv.x -= 0.03f; } } else if (pawnAnimator.headFacing == Rot4.East) { orassanv.x -= 0.1f; } else if (pawnAnimator.headFacing == Rot4.West) { orassanv.x = 0.1f; } else { orassanv.z -= 0.1f; orassanv.y += 1f; if (bodyAddon.bodyPart.Contains("right")) { orassanv.x += 0.05f; } else { orassanv.x -= 0.05f; } } orassanv = orassanv.RotatedBy(pawnAnimator.headAngle); } } GenDraw.DrawMeshNowOrLater(mesh: graphic.MeshAt(rot: headRotInAnimation), loc: headPositionInAnimation + orassanv + (bodyAddon.alignWithHead && !orassan ? headOffset : Vector3.zero),// + v.RotatedBy(Mathf.Acos(Quaternion.Dot(Quaternion.identity, quat)) * 2f * 57.29578f), quat: Quaternion.AngleAxis(angle: num, axis: Vector3.up) * headQuatInAnimation, mat: graphic.MatAt(rot: pawnAnimator.headFacing), drawNow: drawNow);; } else { GenDraw.DrawMeshNowOrLater(mesh, loc, quat, mat, drawNow); } } public static IEnumerable Transpiler(IEnumerable instructions) { List ins = instructions.ToList(); for (int i = 0; i < ins.Count; i++) { Type[] type = new Type[] { typeof(Mesh), typeof(Vector3), typeof(Quaternion), typeof(Material), typeof(bool) }; if (ins[i].OperandIs(AccessTools.Method(typeof(GenDraw), "DrawMeshNowOrLater", type))) { yield return new CodeInstruction(OpCodes.Ldloc, (object)7); //graphic yield return new CodeInstruction(OpCodes.Ldloc, (object)4); //bodyAddon yield return new CodeInstruction(OpCodes.Ldloc, (object)5); //offsetVector/AddonOffset (v) yield return new CodeInstruction(OpCodes.Ldloc, (object)6); //num yield return new CodeInstruction(OpCodes.Ldarg, (object)2); //headOffset yield return new CodeInstruction(OpCodes.Ldarg, (object)3); //pawn yield return new CodeInstruction(OpCodes.Ldarg, (object)0); //renderflags yield return new CodeInstruction(OpCodes.Call, AccessTools.DeclaredMethod(typeof(HarmonyPatch_AlienRace), "RenderHeadAddonInAnimation")); } else { yield return ins[i]; } } } public static bool Prefix(PawnRenderFlags renderFlags, ref Vector3 vector, ref Vector3 headOffset, Pawn pawn, ref Quaternion quat, ref Rot4 rotation) { if(pawn == null) { return true; } CompBodyAnimator anim = pawn.TryGetComp(); if(anim == null) { return true; } if (anim != null && !renderFlags.FlagSet(PawnRenderFlags.Portrait) && anim.isAnimating) { quat = Quaternion.AngleAxis(anim.bodyAngle, Vector3.up); } return true; } } [HarmonyPatch(typeof(PawnGraphicSet), "ResolveApparelGraphics")] public static class HarmonyPatch_ResolveApparelGraphics { public static bool Prefix(ref Pawn ___pawn) { if (___pawn.TryGetComp() != null && ___pawn.TryGetComp().isAnimating) { return false; } return true; } } }