using Cryville.Common; using Cryville.Common.Buffers; using Cryville.Crtr.Components; using System; using System.Collections.Generic; using System.Globalization; using UnityEngine; namespace Cryville.Crtr.Event { public abstract class ContainerHandler { #region Struct public ContainerHandler() { } public abstract string TypeName { get; } /// /// Prehandling , prehandling the events. /// public ContainerState ps; /// /// Backward , disposing events at the backward clipping plane, firing anchor state and temporary state. /// public ContainerState bs; /// /// Current , handling events occuring at the current time. /// public ContainerState cs; /// /// Anchor , computing the start position of the temporary state. /// public ContainerState ns; /// /// Temporary , rendering events between the clipping planes. /// public ContainerState ts; /// /// group, the containing all the generated elements in the . /// protected Transform gogroup; public SkinContext SkinContext; public Vector3 Position { get; protected set; } public Quaternion Rotation { get; protected set; } public bool Alive { get; private set; } public bool Awoken { get; private set; } public bool Disposed { get; private set; } public EventContainer Container { get { return cs.Container; } } SkinContainer skinContainer; protected Judge judge; public void AttachSystems(PdtSkin skin, Judge judge) { skinContainer = new SkinContainer(skin); this.judge = judge; } public readonly Dictionary> Anchors = new Dictionary>(); public readonly Dictionary DynamicAnchors = new Dictionary(); public Anchor OpenedAnchor; protected Anchor a_cur; protected Anchor a_head; protected Anchor a_tail; protected readonly static int _a_cur = IdentifierManager.SharedInstance.Request("cur"); protected readonly static int _a_head = IdentifierManager.SharedInstance.Request("head"); protected readonly static int _a_tail = IdentifierManager.SharedInstance.Request("tail"); public Anchor RegisterAnchor(int name, bool dyn = false, int propSrcCount = 0) { var strname = IdentifierManager.SharedInstance.Retrieve(name); var go = new GameObject("." + strname).transform; go.SetParent(gogroup, false); var result = new Anchor(name, go, propSrcCount); if (dyn) { if (DynamicAnchors.ContainsKey(name)) throw new ArgumentException(string.Format("The anchor \"{0}\" already exists", strname)); DynamicAnchors.Add(name, result); } List list; if (!Anchors.TryGetValue(name, out list)) Anchors.Add(name, list = new List()); list.Add(result); return result; } protected void OpenAnchor(Anchor anchor) { if (OpenedAnchor != null) throw new InvalidOperationException("An anchor has been opened"); OpenedAnchor = anchor; } protected void CloseAnchor() { OpenedAnchor = null; } #endregion #region Logic /// Called upon StartUpdate of ps 17. public virtual void PreInit() { gogroup = new GameObject(TypeName + ":" + Container.GetHashCode().ToString(CultureInfo.InvariantCulture)).transform; SkinContext = new SkinContext(gogroup); if (cs.Parent != null) gogroup.SetParent(cs.Parent.Handler.gogroup, false); a_cur = RegisterAnchor(_a_cur); a_head = RegisterAnchor(_a_head, true); a_tail = RegisterAnchor(_a_tail, true); } public virtual void Init() { skinContainer.MatchStatic(ps); foreach (var i in gogroup.GetComponentsInChildren()) i.Init(); } public virtual void PostInit() { gogroup.gameObject.SetActive(false); } public virtual void Dispose() { if (Disposed) return; Disposed = true; if (gogroup) GameObject.Destroy(gogroup.gameObject); // gogroup.gameObject.SetActive(false); Alive = false; } protected virtual void PreAwake(ContainerState s) { if (gogroup) { gogroup.gameObject.SetActive(true); OpenAnchor(a_head); } Awoken = true; Alive = true; } protected virtual void Awake(ContainerState s) { if (gogroup) CloseAnchor(); } protected virtual void GetPosition(ContainerState s) { } public virtual void StartUpdate(ContainerState s) { if (s.CloneType >= 2 && s.CloneType < 16) { PreAwake(s); Awake(s); } else if (s.CloneType == 17) { Init(); } } public virtual void Update(ContainerState s, StampedEvent ev) { bool flag = !Awoken && s.CloneType >= 2 && s.CloneType < 16; if (flag) PreAwake(s); if (gogroup && s.CloneType <= 2) skinContainer.MatchDynamic(s); if (flag) Awake(s); } public virtual void ExUpdate(ContainerState s, StampedEvent ev) { if (ev is StampedEvent.Anchor) { var tev = (StampedEvent.Anchor)ev; if (gogroup) { OpenAnchor(tev.Target); #if UNITY_5_6_OR_NEWER tev.Target.Transform.SetPositionAndRotation(Position, Rotation); #else tev.Target.Transform.position = GetCurrentWorldPoint(); tev.Target.Transform.rotation = Quaternion.Euler(s.Direction); #endif skinContainer.MatchDynamic(s); CloseAnchor(); } anchorEvPool.Return(tev); } } public virtual void MotionUpdate(byte ct, Chart.Motion ev) { } public virtual void EndUpdate(ContainerState s) { if (s.CloneType < 16) { Awoken = false; if (gogroup && s.CloneType <= 2) { OpenAnchor(a_tail); skinContainer.MatchDynamic(s); CloseAnchor(); } } } #region Anchor public virtual void Anchor() { skinContainer.MatchDynamic(cs); if (cs.Working) PushAnchorEvent(cs.Time, a_cur); static readonly SimpleObjectPool anchorEvPool = new SimpleObjectPool(1024); public void PushAnchorEvent(double time, int name) { Anchor anchor; if (!DynamicAnchors.TryGetValue(name, out anchor)) throw new ArgumentException(string.Format("Specified anchor \"{0}\" not found", IdentifierManager.SharedInstance.Retrieve(name))); PushAnchorEvent(time, anchor); } void PushAnchorEvent(double time, Anchor anchor, int priority = 0, bool forced = false) { var tev = anchorEvPool.Rent(); tev.Time = time; tev.Container = Container; tev.Target = anchor; tev.CanDiscard = !forced; tev.SetPriority(priority); ts.Bus.PushTempEvent(tev); } public virtual void Discard(ContainerState s, StampedEvent ev) { if (ev is StampedEvent.Anchor) { anchorEvPool.Return((StampedEvent.Anchor)ev); } } #endregion #endregion } }