mirror of
https://github.com/misternebula/quantum-space-buddies.git
synced 2025-01-25 15:35:22 +00:00
234 lines
8.8 KiB
C#
234 lines
8.8 KiB
C#
using HarmonyLib;
|
|
using OWML.Common;
|
|
using QSB.Animation.NPC.WorldObjects;
|
|
using QSB.ConversationSync;
|
|
using QSB.Events;
|
|
using QSB.Patches;
|
|
using QSB.Player;
|
|
using QSB.Utility;
|
|
using QSB.WorldSync;
|
|
using System.Linq;
|
|
using System.Reflection;
|
|
using UnityEngine;
|
|
|
|
namespace QSB.Animation.NPC.Patches
|
|
{
|
|
[HarmonyPatch]
|
|
public class CharacterAnimationPatches : QSBPatch
|
|
{
|
|
public override QSBPatchTypes Type => QSBPatchTypes.OnClientConnect;
|
|
|
|
[HarmonyPrefix]
|
|
[HarmonyPatch(typeof(CharacterAnimController), nameof(CharacterAnimController.OnAnimatorIK))]
|
|
public static bool AnimatorIKReplacement(
|
|
CharacterAnimController __instance,
|
|
float ___headTrackingWeight,
|
|
bool ___lookOnlyWhenTalking,
|
|
bool ____inConversation,
|
|
ref float ____currentLookWeight,
|
|
ref Vector3 ____currentLookTarget,
|
|
DampedSpring3D ___lookSpring,
|
|
Animator ____animator,
|
|
CharacterDialogueTree ____dialogueTree)
|
|
{
|
|
if (!WorldObjectManager.AllReady || ConversationManager.Instance == null)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
var playerId = ConversationManager.Instance.GetPlayerTalkingToTree(____dialogueTree);
|
|
var player = QSBPlayerManager.GetPlayer(playerId);
|
|
var qsbObj = QSBWorldSync.GetWorldFromUnity<QSBCharacterAnimController, CharacterAnimController>(__instance); // OPTIMIZE : maybe cache this somewhere... or assess how slow this is
|
|
|
|
PlayerInfo playerToUse = null;
|
|
if (____inConversation)
|
|
{
|
|
if (playerId == uint.MaxValue)
|
|
{
|
|
DebugLog.ToConsole($"Error - {__instance.name} is in conversation with a null player! Defaulting to active camera.", MessageType.Error);
|
|
playerToUse = QSBPlayerManager.LocalPlayer;
|
|
}
|
|
else
|
|
{
|
|
playerToUse = player.CameraBody == null
|
|
? QSBPlayerManager.LocalPlayer
|
|
: player;
|
|
}
|
|
}
|
|
else if (!___lookOnlyWhenTalking && qsbObj.GetPlayersInHeadZone().Count != 0) // IDEA : maybe this would be more fun if characters looked between players at random times? :P
|
|
{
|
|
playerToUse = QSBPlayerManager.GetClosestPlayerToWorldPoint(qsbObj.GetPlayersInHeadZone(), __instance.transform.position);
|
|
}
|
|
else if (QSBPlayerManager.PlayerList.Count != 0)
|
|
{
|
|
playerToUse = QSBPlayerManager.GetClosestPlayerToWorldPoint(__instance.transform.position, true);
|
|
}
|
|
|
|
var localPosition = playerToUse != null
|
|
? ____animator.transform.InverseTransformPoint(playerToUse.CameraBody.transform.position)
|
|
: Vector3.zero;
|
|
|
|
var targetWeight = ___headTrackingWeight;
|
|
if (___lookOnlyWhenTalking)
|
|
{
|
|
if (!____inConversation
|
|
|| qsbObj.GetPlayersInHeadZone().Count == 0
|
|
|| !qsbObj.GetPlayersInHeadZone().Contains(playerToUse))
|
|
{
|
|
targetWeight *= 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (qsbObj.GetPlayersInHeadZone().Count == 0
|
|
|| !qsbObj.GetPlayersInHeadZone().Contains(playerToUse))
|
|
{
|
|
targetWeight *= 0;
|
|
}
|
|
}
|
|
|
|
____currentLookWeight = Mathf.Lerp(____currentLookWeight, targetWeight, Time.deltaTime * 2f);
|
|
____currentLookTarget = ___lookSpring.Update(____currentLookTarget, localPosition, Time.deltaTime);
|
|
____animator.SetLookAtPosition(____animator.transform.TransformPoint(____currentLookTarget));
|
|
____animator.SetLookAtWeight(____currentLookWeight);
|
|
return false;
|
|
|
|
}
|
|
|
|
[HarmonyPrefix]
|
|
[HarmonyPatch(typeof(SolanumAnimController), nameof(SolanumAnimController.LateUpdate))]
|
|
public static bool SolanumLateUpdateReplacement(SolanumAnimController __instance)
|
|
{
|
|
if (__instance._animatorStateEvents == null)
|
|
{
|
|
__instance._animatorStateEvents = __instance._animator.GetBehaviour<AnimatorStateEvents>();
|
|
__instance._animatorStateEvents.OnEnterState += __instance.OnEnterAnimatorState;
|
|
}
|
|
|
|
var qsbObj = QSBWorldSync.GetWorldFromUnity<QSBSolanumAnimController, SolanumAnimController>(__instance);
|
|
var playersInHeadZone = qsbObj.GetPlayersInHeadZone();
|
|
|
|
Transform targetCamera = playersInHeadZone == null || playersInHeadZone.Count == 0
|
|
? __instance._playerCameraTransform
|
|
: QSBPlayerManager.GetClosestPlayerToWorldPoint(playersInHeadZone, __instance.transform.position).CameraBody.transform;
|
|
|
|
var targetValue = Quaternion.LookRotation(targetCamera.position - __instance._headBoneTransform.position, __instance.transform.up);
|
|
__instance._currentLookRotation = __instance._lookSpring.Update(__instance._currentLookRotation, targetValue, Time.deltaTime);
|
|
|
|
var position = __instance._headBoneTransform.position + (__instance._currentLookRotation * Vector3.forward);
|
|
__instance._localLookPosition = __instance.transform.InverseTransformPoint(position);
|
|
|
|
return false;
|
|
}
|
|
|
|
[HarmonyPrefix]
|
|
[HarmonyPatch(typeof(CharacterAnimController), nameof(CharacterAnimController.OnZoneExit))]
|
|
public static bool HeadZoneExit(CharacterAnimController __instance)
|
|
{
|
|
var qsbObj = QSBWorldSync.GetWorldFromUnity<QSBCharacterAnimController, CharacterAnimController>(__instance);
|
|
QSBEventManager.FireEvent(EventNames.QSBExitNonNomaiHeadZone, qsbObj.ObjectId);
|
|
return false;
|
|
}
|
|
|
|
[HarmonyPrefix]
|
|
[HarmonyPatch(typeof(CharacterAnimController), nameof(CharacterAnimController.OnZoneEntry))]
|
|
public static bool HeadZoneEntry(CharacterAnimController __instance)
|
|
{
|
|
var qsbObj = QSBWorldSync.GetWorldFromUnity<QSBCharacterAnimController, CharacterAnimController>(__instance);
|
|
QSBEventManager.FireEvent(EventNames.QSBEnterNonNomaiHeadZone, qsbObj.ObjectId);
|
|
return false;
|
|
}
|
|
|
|
[HarmonyPrefix]
|
|
[HarmonyPatch(typeof(NomaiConversationManager), nameof(NomaiConversationManager.OnEnterWatchVolume))]
|
|
public static bool EnterWatchZone(NomaiConversationManager __instance)
|
|
{
|
|
var qsbObj = QSBWorldSync.GetWorldFromUnity<QSBSolanumAnimController, SolanumAnimController>(__instance._solanumAnimController);
|
|
QSBEventManager.FireEvent(EventNames.QSBEnterNomaiHeadZone, qsbObj.ObjectId);
|
|
return false;
|
|
}
|
|
|
|
[HarmonyPrefix]
|
|
[HarmonyPatch(typeof(NomaiConversationManager), nameof(NomaiConversationManager.OnExitWatchVolume))]
|
|
public static bool ExitWatchZone(NomaiConversationManager __instance)
|
|
{
|
|
var qsbObj = QSBWorldSync.GetWorldFromUnity<QSBSolanumAnimController, SolanumAnimController>(__instance._solanumAnimController);
|
|
QSBEventManager.FireEvent(EventNames.QSBExitNomaiHeadZone, qsbObj.ObjectId);
|
|
return false;
|
|
}
|
|
|
|
[HarmonyPrefix]
|
|
[HarmonyPatch(typeof(FacePlayerWhenTalking), nameof(FacePlayerWhenTalking.OnStartConversation))]
|
|
public static bool OnStartConversation(FacePlayerWhenTalking __instance)
|
|
{
|
|
var playerId = ConversationManager.Instance.GetPlayerTalkingToTree(__instance._dialogueTree);
|
|
if (playerId == uint.MaxValue)
|
|
{
|
|
DebugLog.ToConsole($"Error - No player talking to {__instance._dialogueTree.name}!", MessageType.Error);
|
|
return false;
|
|
}
|
|
|
|
var player = QSBPlayerManager.GetPlayer(playerId);
|
|
|
|
var distance = player.Body.transform.position - __instance.transform.position;
|
|
var vector2 = distance - Vector3.Project(distance, __instance.transform.up);
|
|
var angle = Vector3.Angle(__instance.transform.forward, vector2) * Mathf.Sign(Vector3.Dot(vector2, __instance.transform.right));
|
|
var axis = __instance.transform.parent.InverseTransformDirection(__instance.transform.up);
|
|
var lhs = Quaternion.AngleAxis(angle, axis);
|
|
__instance.FaceLocalRotation(lhs * __instance.transform.localRotation);
|
|
|
|
return false;
|
|
}
|
|
|
|
[HarmonyPrefix]
|
|
[HarmonyPatch(typeof(CharacterDialogueTree), nameof(CharacterDialogueTree.StartConversation))]
|
|
public static bool StartConversation(CharacterDialogueTree __instance)
|
|
{
|
|
var allNpcAnimControllers = QSBWorldSync.GetWorldObjects<INpcAnimController>();
|
|
var ownerOfThis = allNpcAnimControllers.FirstOrDefault(x => x.GetDialogueTree() == __instance);
|
|
if (ownerOfThis == default)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
var id = QSBWorldSync.GetIdFromTypeSubset(ownerOfThis);
|
|
QSBEventManager.FireEvent(EventNames.QSBNpcAnimEvent, AnimationEvent.StartConversation, id);
|
|
return true;
|
|
}
|
|
|
|
[HarmonyPrefix]
|
|
[HarmonyPatch(typeof(CharacterDialogueTree), nameof(CharacterDialogueTree.EndConversation))]
|
|
public static bool EndConversation(CharacterDialogueTree __instance)
|
|
{
|
|
var allNpcAnimControllers = QSBWorldSync.GetWorldObjects<INpcAnimController>();
|
|
var ownerOfThis = allNpcAnimControllers.FirstOrDefault(x => x.GetDialogueTree() == __instance);
|
|
if (ownerOfThis == default)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
var id = QSBWorldSync.GetIdFromTypeSubset(ownerOfThis);
|
|
QSBEventManager.FireEvent(EventNames.QSBNpcAnimEvent, AnimationEvent.EndConversation, id);
|
|
return true;
|
|
}
|
|
|
|
[HarmonyPrefix]
|
|
[HarmonyPatch(typeof(KidRockController), nameof(KidRockController.Update))]
|
|
public static bool UpdateReplacement(KidRockController __instance)
|
|
{
|
|
if (!WorldObjectManager.AllReady)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
var qsbObj = QSBWorldSync.GetWorldObjects<QSBCharacterAnimController>().First(x => x.GetDialogueTree() == __instance._dialogueTree);
|
|
|
|
if (!__instance._throwingRock && !qsbObj.InConversation() && Time.time > __instance._nextThrowTime)
|
|
{
|
|
__instance.StartRockThrow();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
}
|
|
} |