Liberi
An exergame built for kids with CP!
Vitals.cs
1 using UnityEngine;
2 using System.Collections;
3 using System.Collections.Generic;
4 using Janus;
5 using System;
6 using System.Linq;
7 using System.Reflection;
8 
9 public delegate void VitalsHitHandler (Vitals vitals, HitInfo hit);
10 public delegate void VitalsKilledHandler (Vitals vitals, HitInfo fatalHit);
11 public delegate void VitalsRevivedHandler (Vitals vitals);
12 
16 [AddComponentMenu("Liberi/Vitals")]
17 public class Vitals : MonoBehaviour
18 {
22  public event VitalsHitHandler Hit = delegate {};
26  public event VitalsKilledHandler Killed = delegate {};
30  public event VitalsRevivedHandler Revived = delegate {};
31 
36  public bool IsInvulnerable
37  {
38  get { return !_isVulnerable.LastValue; }
39  set { _isVulnerable[0] = !value; }
40  }
41 
46  public bool IsVulnerable
47  {
48  get { return _isVulnerable.LastValue; }
49  set { _isVulnerable[0] = value; }
50  }
51 
55  public float CurrentHealth
56  {
57  get { return Health.IsEmpty ? MaxHealth.LastValue : Health.LastValue; }
58  set { Health[0] = Mathf.Clamp(value, 0, MaxHealth.LastValue); }
59  }
60 
64  public float CurrentNormalizedHealth
65  {
66  get
67  {
68  if (MaxHealth.LastValue == 0)
69  return 0;
70  else
71  return this.CurrentHealth / MaxHealth.LastValue;
72  }
73  set { this.CurrentHealth = value * MaxHealth.LastValue; }
74  }
75 
79  public bool IsAlive
80  {
81  get { return this.CurrentHealth > 0f; }
82  }
83 
87  public bool IsDead
88  {
89  get { return this.CurrentHealth <= 0f; }
90  }
91 
95  public float BaseMaxHealth = 100f;
99  public string VitalTag;
103  public bool HealthRegenEnabled = false;
107  public float HealthRegen = 0f;
111  public float HealthRegenInterval = .5f;
115  public float HitTimeout = 0f;
119  public bool DespawnWhenKilled = false;
128 
132  public Timeline<float> Health;
136  public Timeline<float> MaxHealth;
137 
138  Timeline<bool> _isVulnerable;
139  Timeline<GameMessage> _hits;
140  Blinker _blinker;
141  float _lastHitTime;
142 
143  void Awake ()
144  {
145  _blinker = GetComponent<Blinker>();
146 
147  _isVulnerable = Sync.CreateDiscreteState<bool>();
148 
149  Health = Sync.CreateDiscreteState<float>();
150  Health.CacheSize = 1;
151 
152  MaxHealth = Sync.CreateDiscreteState<float>();
153  MaxHealth.CacheSize = 1;
154 
155  _hits = Sync.CreateEvent<GameMessage>();
156  }
157 
158  void OnSpawn ()
159  {
160  _hits.EntryPassed += OnHitMessage;
161  Health.EntryPassed += OnHealthChange;
162 
163  if (Sync.IsLocal(gameObject))
164  {
165  _isVulnerable[0] = true;
166  MaxHealth[0] = BaseMaxHealth;
167  Health[0] = MaxHealth.LastValue;
168 
169  StartCoroutine(RegenHealth());
170  }
171  }
172 
173  void OnHealthChange (Timeline<float> timeline, TimelineEntry<float> entry)
174  {
175  if (entry.Prev != null && entry.Prev.Value <= 0 && entry.Value > 0)
176  Revived(this);
177 
178  if (entry.Value <= 0 && DespawnWhenKilled)
179  Sync.Despawn(this);
180  }
181 
192  public static bool ApplyHit (Component targetObject, Vector3 location, float value = 0f,
193  GameObject sourceObject = null, string hitTag = "", UJeli details = null)
194  {
195  return ApplyHit(targetObject.gameObject, location, value, sourceObject, hitTag, details);
196  }
197 
208  public static bool ApplyHit (Component targetObject, Vector3 location, float value = 0f,
209  Component sourceObject = null, string hitTag = "", UJeli details = null)
210  {
211  return ApplyHit(targetObject.gameObject, location, value, sourceObject.gameObject, hitTag, details);
212  }
213 
224  public static bool ApplyHit (GameObject targetObject, Vector3 location, float value = 0f,
225  Component sourceObject = null, string hitTag = "", UJeli details = null)
226  {
227  return ApplyHit(targetObject, location, value, sourceObject.gameObject, hitTag, details);
228  }
229 
240  public static bool ApplyHit (GameObject targetObject, Vector3 location, float value = 0f,
241  GameObject sourceObject = null, string hitTag = "", UJeli details = null)
242  {
243  var vitals = targetObject.GetComponent<Vitals>();
244 
245  if (vitals != null)
246  {
247  while (vitals.RedirectHitsTo != null)
248  {
249  vitals = vitals.RedirectHitsTo;
250  }
251 
252  if (!vitals._isVulnerable.LastValue)
253  return false;
254 
255  vitals.PropagateHit(false, location, value, false, sourceObject, hitTag, details);
256 
257  return true;
258  }
259 
260  return false;
261  }
262 
263  void PropagateHit (bool isFinal, Vector3 location, float value, bool isFatal, GameObject sourceObject, string hitTag, UJeli details)
264  {
265  int sourceObjectIndex = sourceObject != null ? Sync.GetObjectIndex(sourceObject) : -1;
266 
267  string detailsString = details != null ? details.ToString() : "";
268 
269  var hitMsg = new GameMessage();
270  hitMsg.Write(isFinal);
271  hitMsg.Write(location);
272  hitMsg.Write(value);
273  hitMsg.Write(isFatal);
274  hitMsg.Write(sourceObjectIndex);
275  hitMsg.Write(hitTag);
276  hitMsg.Write(detailsString);
277 
278  _hits[0] = hitMsg;
279  }
280 
281  void Update ()
282  {
283  if (HealthMeter != null)
284  HealthMeter.Value = CurrentNormalizedHealth;
285  }
286 
287  IEnumerator RegenHealth ()
288  {
289  while (true)
290  {
291  if (HealthRegenEnabled && Health.LastValue < MaxHealth.LastValue)
292  this.CurrentHealth += HealthRegen * HealthRegenInterval;
293 
294  yield return new WaitForSeconds(HealthRegenInterval);
295  }
296  }
297 
298  void OnHitMessage (Timeline<GameMessage> timeline, TimelineEntry<GameMessage> entry)
299  {
300  var hitMsg = entry.Value;
301 
302  bool isFinal = hitMsg.ReadBoolean();
303  Vector3 location = hitMsg.ReadVector3();
304  float value = hitMsg.ReadFloat();
305  bool isFatal = hitMsg.ReadBoolean();
306  int sourceObjectIndex = hitMsg.ReadInt32();
307  GameObject sourceObject = sourceObjectIndex != -1 ? Sync.GetObject(sourceObjectIndex) : null;
308  string hitTag = hitMsg.ReadString();
309  string detailsString = hitMsg.ReadString();
310  UJeli details = UJeli.Parse(detailsString);
311  hitMsg.FinishReading();
312 
313  if (sourceObjectIndex != -1 && sourceObject == null)
314  return;
315 
316  var hit = new HitInfo()
317  {
318  TargetVitals = this,
319  Location = location,
320  IsFatal = isFatal,
321  Value = value,
322  SourceObject = sourceObject,
323  HitTag = hitTag,
324  Details = details
325  };
326 
327  if (isFinal)
328  {
329  if (_blinker != null)
330  _blinker.Blink(HitTimeout);
331 
332  Sync.SendEnabledMessage(this, "OnHit", hit);
333 
334  Hit(this, hit);
335 
336  if (hit.SourceObject != null)
337  Sync.SendEnabledMessage(hit.SourceObject, "OnOutgoingHit", gameObject, hit);
338 
339  if (hit.IsFatal)
340  {
341  Killed(this, hit);
342 
343  if (Sync.IsServer)
344  return;
345 
346  GameObject shooter = null;
347 
348  if (hit.Details.HasChild("Shooter"))
349  shooter = hit.Details["Shooter"].GameObjectValue;
350 
351  var trueSourceObject = shooter ?? hit.SourceObject;
352 
353  if (Sync.IsLocal(this))
354  {
355  // This is a local Vitals dying (ie, a player's avatar)
356  // If it's an Avatar killing this, log the Nickname of the avatar owner, else log the object's name
357  if (trueSourceObject.CompareTag("Avatar"))
358  {
359  string playerName = Game.GetPlayerNickname(Sync.GetOwner(trueSourceObject));
360  Study.LogEvent(VitalsLogEvent.KilledBy, playerName);
361  }
362  else
363  Study.LogEvent(VitalsLogEvent.KilledBy, trueSourceObject.name);
364  }
365  else if (Sync.IsLocal(trueSourceObject))
366  {
367  // The sourceObject or shooter is local (ie, the player killed something)
368  // If it's an Avatar being killed, log the Nickname of the victim, else log the object's name
369  if (gameObject.CompareTag("Avatar"))
370  {
371  string playerName = Game.GetPlayerNickname(Sync.GetOwner(this));
372  Study.LogEvent(VitalsLogEvent.Killed, playerName);
373  }
374  else
375  Study.LogEvent(VitalsLogEvent.Killed, name);
376  }
377 
378 
379  }
380  }
381  else
382  {
383  if (Sync.IsLocal(gameObject) && _isVulnerable.LastValue)
384  {
385  if (Time.time - _lastHitTime > HitTimeout)
386  {
387  Sync.SendEnabledMessage(this, "PreHit", hit);
388 
389  if (!hit.Cancel)
390  {
391  if (hit.Value != 0)
392  {
393  bool wasAlive = this.IsAlive;
394 
395  Health[0] = Mathf.Clamp(Health.LastValue - hit.Value, 0, MaxHealth.LastValue);
396 
397  if (wasAlive && !this.IsAlive)
398  hit.IsFatal = true;
399  }
400 
401  _lastHitTime = Time.time;
402 
403  PropagateHit(true, location, value, hit.IsFatal, sourceObject, hitTag, details);
404  }
405  }
406  }
407  }
408  }
409 }
bool IsAlive
Whether or not the object's health is currently over zero.
Definition: Vitals.cs:80
float HitTimeout
The length of time after being hit that this object is invulnerable.
Definition: Vitals.cs:115
static void Despawn(GameObject go)
Despawn the given object.
Definition: Sync.Static.cs:779
VitalsKilledHandler Killed
Event fired whenever this object's health reaches zero.
Definition: Vitals.cs:26
static bool ApplyHit(Component targetObject, Vector3 location, float value=0f, Component sourceObject=null, string hitTag="", UJeli details=null)
Apply a hit to a given object.
Definition: Vitals.cs:208
Describes a hit on an object.
Definition: HitInfo.cs:12
Timeline< float > Health
The current health this object has.
Definition: Vitals.cs:132
Manages the "vitals" of an object (health, regeneration, damage).
Definition: Vitals.cs:17
string VitalTag
The vital "tag" of this object. Used by Hitter to selectively apply damage.
Definition: Vitals.cs:99
void Blink(float duration)
Blink the object for the given duration.
Definition: Blinker.cs:38
float Value
A value between 0 and 1 at which this bar is filled.
Definition: MeterSticker.cs:14
float HealthRegen
The rate at which health is regenerated, in health units per second.
Definition: Vitals.cs:107
static bool ApplyHit(GameObject targetObject, Vector3 location, float value=0f, Component sourceObject=null, string hitTag="", UJeli details=null)
Apply a hit to a given object.
Definition: Vitals.cs:224
static bool IsServer
Gets whether or not this peer is the server.
Definition: Sync.Static.cs:53
float BaseMaxHealth
The base max health this object is allowed to have (before any boosts/penalties are applied)...
Definition: Vitals.cs:95
static void SendEnabledMessage(GameObject go, string message, params object[] args)
Invokes a method on all enabled scripts in the GameObject.
Definition: Sync.Static.cs:740
bool IsDead
Whether or not the object's health is currently zero.
Definition: Vitals.cs:88
static bool IsLocal(GameObject go)
Gets whether or not the given object is owned by the local peer.
Definition: Sync.Static.cs:290
bool HealthRegenEnabled
Whether or not health regeneration is enabled.
Definition: Vitals.cs:103
Vitals RedirectHitsTo
An optional Vitals component to which all incoming hits to this component are redirected.
Definition: Vitals.cs:127
MeterSticker HealthMeter
An optional health meter to render the health.
Definition: Vitals.cs:123
static bool ApplyHit(GameObject targetObject, Vector3 location, float value=0f, GameObject sourceObject=null, string hitTag="", UJeli details=null)
Apply a hit to a given object.
Definition: Vitals.cs:240
float CurrentNormalizedHealth
The current health of the object as a ratio of its maximum health.
Definition: Vitals.cs:65
static int GetOwner(GameObject go)
Gets the owner peer index of a given object.
Definition: Sync.Static.cs:263
bool IsInvulnerable
Whether or not the object is currently invulnerable. Note that this is not affected by the hit timeou...
Definition: Vitals.cs:37
VitalsRevivedHandler Revived
Event fired whenever this object's health changes from zero to non-zero.
Definition: Vitals.cs:30
static int GetObjectIndex(GameObject go)
Gets the object index of a given object.
Definition: Sync.Static.cs:190
bool DespawnWhenKilled
Whether or not to automatically despawn this object when its health reaches zero. ...
Definition: Vitals.cs:119
A class for serializing game data into a stream for propagation between objects and peers...
Definition: GameMessage.cs:14
float HealthRegenInterval
The interval between each health regeneration tick, i.e. the granularity of regeneration.
Definition: Vitals.cs:111
VitalsHitHandler Hit
Event fired whenever this object is hit.
Definition: Vitals.cs:22
Timeline< float > MaxHealth
The maximum health this object is allowed to have.
Definition: Vitals.cs:136
Class for managing study-related game components.
Definition: Study.cs:13
static bool ApplyHit(Component targetObject, Vector3 location, float value=0f, GameObject sourceObject=null, string hitTag="", UJeli details=null)
Apply a hit to a given object.
Definition: Vitals.cs:192
static GameObject GetObject(int objectIndex)
Get an object by its object index.
Definition: Sync.Static.cs:143
A Sticker for showing health bars.
Definition: MeterSticker.cs:9
bool IsVulnerable
Whether or not the object is currently vulnerable. Note that this is not affected by the hit timeout...
Definition: Vitals.cs:47
float CurrentHealth
The current health of the object.
Definition: Vitals.cs:56
Blinks the renderers in an object on and off. Good for damage effects.
Definition: Blinker.cs:14
Unity version of Jeli markup class.
Definition: UJeli.cs:10
static void LogEvent(Enum eventType, params object[] args)
Log a game event.
Definition: Study.cs:57
This class server two main functions: 1) As a MonoBehaviour, it allows for network synchronization of...
Definition: Sync.cs:13