using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; using RimWorld; using rjw.Modules.Interactions.Helpers; using rjw.Modules.Interactions.Objects; using UnityEngine; using Verse; using Verse.AI; using rjw.Modules.Interactions.Enums; namespace Rimworld_Animations { public static class AnimationUtility { /* Note: always make the list in this order: Female pawns, animal female pawns, male pawns, animal male pawns */ public static AnimationDef tryFindAnimation(ref List participants, rjw.xxx.rjwSextype sexType = 0, rjw.SexProps sexProps = null) { InteractionWithExtension interaction = InteractionHelper.GetWithExtension(sexProps.dictionaryKey); participants = participants.OrderBy(p => p.jobs.curDriver is rjw.JobDriver_Sex && !(p.jobs.curDriver as rjw.JobDriver_Sex).Sexprops.isRevese /*Gargulefix!!!*/ && !(p.jobs.curDriver as rjw.JobDriver_Sex).Sexprops.isReceiver) .OrderBy(p => rjw.xxx.can_fuck(p)) .ToList(); List localParticipants = new List(participants); IEnumerable options = DefDatabase.AllDefs.Where((AnimationDef x) => { if (x.actors.Count != localParticipants.Count) { if (AnimationSettings.debugMode) Log.Message(x.defName.ToStringSafe() + " not selected -- doesn't match count"); return false; } for (int i = 0; i < x.actors.Count; i++) { if (rjw.RJWPreferenceSettings.Malesex == rjw.RJWPreferenceSettings.AllowedSex.Nohomo) { if (rjw.xxx.is_male(localParticipants[i]) && x.actors[i].isFucked) { if (AnimationSettings.debugMode) Log.Message(x.defName.ToStringSafe() + " not selected -- " + localParticipants[i].def.defName.ToStringSafe() + " is fucked, shouldn't be due to homo rules"); return false; } } if (x.actors[i].requiredGender != null && !x.actors[i].requiredGender.Contains(localParticipants[i].gender.ToStringSafe())) { if (AnimationSettings.debugMode) { Log.Message(string.Concat(new string[] { x.defName.ToStringSafe(), " not selected -- ", localParticipants[i].def.defName.ToStringSafe(), " ", localParticipants[i].Name.ToStringSafe(), " does not match required gender" })); } return false; } if ((x.actors[i].blacklistedRaces != null) && x.actors[i].blacklistedRaces.Contains(localParticipants[i].def.defName)) { if(AnimationSettings.debugMode) Log.Message(x.defName.ToStringSafe() + " not selected -- " + localParticipants[i].def.defName.ToStringSafe() + " " + localParticipants[i].Name.ToStringSafe() + " is blacklisted"); return false; } if(x.actors[i].defNames.Contains("Human")) { if (!rjw.xxx.is_human(localParticipants[i])) { if (AnimationSettings.debugMode) Log.Message(x.defName.ToStringSafe() + " not selected -- " + localParticipants[i].def.defName.ToStringSafe() + " " + localParticipants[i].Name.ToStringSafe() + " is not human"); return false; } } else if (!x.actors[i].bodyDefTypes.Contains(localParticipants[i].RaceProps.body)) { if (!x.actors[i].defNames.Contains(localParticipants[i].def.defName)) { if (AnimationSettings.debugMode) { string animInfo = x.defName.ToStringSafe() + " not selected -- " + localParticipants[i].def.defName.ToStringSafe() + " " + localParticipants[i].Name.ToStringSafe() + " is not "; foreach(String defname in x.actors[i].defNames) { animInfo += defname + ", "; } if (AnimationSettings.debugMode) Log.Message(animInfo); } return false; } } //genitals checking if(!GenitalCheckForPawn(x.actors[i].requiredGenitals, localParticipants[i], out string failReason)) { if (AnimationSettings.debugMode) Log.Message("Didn't select " + x.defName + ", " + localParticipants[i].Name + " " + failReason); return false; } //TESTING ANIMATIONS ONLY REMEMBER TO COMMENT OUT BEFORE PUSH /* if (x.defName != "Cunnilingus") return false; */ if (x.actors[i].isFucking && !rjw.xxx.can_fuck(localParticipants[i])) { if (AnimationSettings.debugMode) Log.Message(x.defName.ToStringSafe() + " not selected -- " + localParticipants[i].def.defName.ToStringSafe() + " " + localParticipants[i].Name.ToStringSafe() + " can't fuck"); return false; } if (x.actors[i].isFucked && !rjw.xxx.can_be_fucked(localParticipants[i])) { if (AnimationSettings.debugMode) Log.Message(x.defName.ToStringSafe() + " not selected -- " + localParticipants[i].def.defName.ToStringSafe() + " " + localParticipants[i].Name.ToStringSafe() + " can't be fucked"); return false; } } if (AnimationSettings.debugMode) Log.Message(x.defName.ToStringSafe() + " selected"); return true; }); List optionsWithInteractionType = options.ToList().FindAll(x => x.interactionDefTypes != null && x.interactionDefTypes.Contains(sexProps.sexType.ToStringSafe())); if (optionsWithInteractionType.Any()) { if (AnimationSettings.debugMode) Log.Message("Selecting animation for interaction type " + sexProps.sexType.ToStringSafe() + "..."); return optionsWithInteractionType.RandomElement(); } List optionsWithSexType = options.ToList().FindAll(x => x.sexTypes != null && x.sexTypes.Contains(sexType)); if (optionsWithSexType.Any()) { if (AnimationSettings.debugMode) Log.Message("Selecting animation for rjwSexType " + sexType.ToStringSafe() + "..."); return optionsWithSexType.RandomElement(); } /* if(optionsWithInitiator.Any()) { if (AnimationSettings.debugMode) Log.Message("Selecting animation for initiators..."); } */ if (options != null && options.Any()) { if (AnimationSettings.debugMode) Log.Message("Randomly selecting animation..."); return options.RandomElement(); } else return null; } public static void RenderPawnHeadMeshInAnimation1(Mesh mesh, Vector3 loc, Quaternion quaternion, Material material, bool drawNow, Pawn pawn) { if (pawn == null || pawn.Map != Find.CurrentMap) { GenDraw.DrawMeshNowOrLater(mesh, loc, quaternion, material, drawNow); return; } if (!CompBodyAnimator.IsAnimating(pawn)) { GenDraw.DrawMeshNowOrLater(mesh, loc, quaternion, material, drawNow); } else { CompBodyAnimator pawnAnimator = pawn.TryGetComp(); Vector3 pawnHeadPosition = pawnAnimator.getPawnHeadPosition(); pawnHeadPosition.y = loc.y; GenDraw.DrawMeshNowOrLater(MeshPool.humanlikeHeadSet.MeshAt(pawnAnimator.headFacing), pawnHeadPosition, Quaternion.AngleAxis(pawnAnimator.headAngle, Vector3.up), material, true); } } public static void AdjustHead(ref Quaternion quat, ref Rot4 bodyFacing, ref Vector3 pos, ref float angle, Pawn pawn, PawnRenderFlags flags) { if (flags.FlagSet(PawnRenderFlags.Portrait)) return; if (CompBodyAnimator.IsAnimating(pawn)) { CompBodyAnimator anim = pawn.TryGetComp(); bodyFacing = anim.headFacing; angle = anim.headAngle; quat = Quaternion.AngleAxis(anim.headAngle, Vector3.up); pos = anim.getPawnHeadOffset(); } } public static void RenderPawnHeadMeshInAnimation(Mesh mesh, Vector3 loc, Quaternion quaternion, Material material, bool portrait, Pawn pawn, float bodySizeFactor = 1) { if (pawn == null) { GenDraw.DrawMeshNowOrLater(mesh, loc, quaternion, material, portrait); return; } if (!CompBodyAnimator.IsAnimating(pawn) || portrait) { GenDraw.DrawMeshNowOrLater(mesh, loc, quaternion, material, portrait); } else { CompBodyAnimator pawnAnimator = pawn.TryGetComp(); Vector3 pawnHeadPosition = pawnAnimator.getPawnHeadPosition(); pawnHeadPosition.x *= bodySizeFactor; pawnHeadPosition.x *= bodySizeFactor; pawnHeadPosition.y = loc.y; GenDraw.DrawMeshNowOrLater(mesh, pawnHeadPosition, Quaternion.AngleAxis(pawnAnimator.headAngle, Vector3.up), material, portrait); } } public static bool GenitalCheckForPawn(List requiredGenitals, Pawn pawn, out string failReason) { failReason = null; if (requiredGenitals != null) { if (requiredGenitals.Contains("Vagina")) { if (!rjw.Genital_Helper.has_vagina(pawn)) { failReason = "missing vagina"; return false; } } if (requiredGenitals.Contains("Penis")) { if (!(rjw.Genital_Helper.has_multipenis(pawn) || rjw.Genital_Helper.has_penis_infertile(pawn) || rjw.Genital_Helper.has_penis_fertile(pawn) || rjw.Genital_Helper.has_ovipositorM(pawn) || rjw.Genital_Helper.has_ovipositorF(pawn))) { failReason = "missing penis"; return false; } } if (requiredGenitals.Contains("Mouth")) { if (!rjw.Genital_Helper.has_mouth(pawn)) { failReason = "missing mouth"; return false; } } if (requiredGenitals.Contains("Anus")) { if (!rjw.Genital_Helper.has_anus(pawn)) { failReason = "missing anus"; return false; } } if (requiredGenitals.Contains("Breasts")) { if (!rjw.Genital_Helper.can_do_breastjob(pawn)) { failReason = "missing breasts"; return false; } } if (requiredGenitals.Contains("NoVagina")) { if (rjw.Genital_Helper.has_vagina(pawn)) { failReason = "has vagina"; return false; } } if (requiredGenitals.Contains("NoPenis")) { if ((rjw.Genital_Helper.has_multipenis(pawn) || rjw.Genital_Helper.has_penis_infertile(pawn) || rjw.Genital_Helper.has_penis_fertile(pawn))) { failReason = "has penis"; return false; } } if (requiredGenitals.Contains("NoMouth")) { if (rjw.Genital_Helper.has_mouth(pawn)) { failReason = "has mouth"; return false; } } if (requiredGenitals.Contains("NoAnus")) { if (rjw.Genital_Helper.has_anus(pawn)) { failReason = "has anus"; return false; } } if (requiredGenitals.Contains("NoBreasts")) { if (rjw.Genital_Helper.can_do_breastjob(pawn)) { failReason = "has breasts"; return false; } } } return true; } public static Rot4 PawnHeadRotInAnimation(Pawn pawn, Rot4 regularPos) { Debug.Log("Test"); if(pawn?.TryGetComp() != null && pawn.TryGetComp().isAnimating) { return pawn.TryGetComp().headFacing; } return regularPos; } } }