Liberi
An exergame built for kids with CP!
Character.cs
1 using UnityEngine;
2 using System;
3 using System.Collections;
4 using System.Collections.Generic;
5 using System.Linq;
6 using Janus;
7 
11 [AddComponentMenu("Liberi/Character")]
12 public class Character: MonoBehaviour
13 {
14  public bool IsSkillReady
15  {
16  get
17  {
18  return CanUseSkill && _energy >= 1f;
19  }
20  }
21 
22  public Persona ActivePersona
23  {
24  get
25  {
26  int index = _activePersonaIndex.LastValue;
27 
28  if (index < 0 || index >= Personas.Length)
29  return null;
30 
31  return Personas[index];
32  }
33  }
34 
35  public string ActivePersonaID
36  {
37  get
38  {
39  var activePersona = this.ActivePersona;
40 
41  if (activePersona == null)
42  return null;
43 
44  return activePersona.ID;
45  }
46  }
47 
48  public bool IsTurbo
49  {
50  get { return _isTurbo.LastValue; }
51  set
52  {
53  if (!Sync.IsLocal(this))
54  return;
55 
56  _isTurbo[0] = value;
57  }
58  }
59 
60  public float Energy
61  {
62  get { return _energy; }
63  }
64 
65  const float _personaChangeCooldown = .5f;
66 
67  [NonSerialized]
68  public bool CanUseSkill = true;
69  public bool CanChargeEnergy = true;
70  public CharacterTrait[] Traits;
71  public Persona[] Personas;
72  [SpawnDetail]
73  public float BotTurboWarmup = 60f;
74 
75  Timeline<int> _activePersonaIndex;
76  Timeline<bool> _isTurbo;
77  Timeline<bool> _skillUses;
78  float _energy;
79  bool _isBot;
80 
81  void Awake ()
82  {
83  _activePersonaIndex = Sync.CreateDiscreteState<int>();
84  _isTurbo = Sync.CreateDiscreteState<bool>();
85  _skillUses = Sync.CreateEvent<bool>();
86 
87  foreach (var trait in Traits)
88  {
89  trait.Character = this;
90  }
91 
92  foreach (var persona in Personas)
93  {
94  persona.Character = this;
95  }
96  }
97 
98  void OnSpawn (UJeli details)
99  {
100  _activePersonaIndex.EntryPassed += OnActivePersonaIndexMessage;
101  _isTurbo.EntryPassed += OnTurboMessage;
102  _skillUses.EntryPassed += OnSkillUseMessage;
103 
104  if (Sync.IsClientOwned(this))
105  {
106  var characterProfile = Game.GetPlayerProfile(Sync.GetOwner(this)).GetCharacter(name);
107 
108  if (Sync.IsLocal(this))
109  {
110  if (characterProfile != null)
111  SetActivePersona(characterProfile.LastActivePersonaID);
112 
113  StartCoroutine(UpdateHumanPersonaAsync());
114  StartCoroutine(UpdateHumanEnergyAsync());
115  StartCoroutine(UpdateHumanTurboAsync());
116 
118 
119  foreach (var trait in Traits)
120  {
121  int traitPoints = 0;
122  characterProfile.TraitPoints.TryGetValue(trait.ID, out traitPoints);
123  trait.Points = traitPoints;
124  }
125  }
126  }
127  else
128  {
129  if (Sync.IsLocal(this))
130  {
131  UJeli personaJeli = details["Persona"];
132 
133  SetActivePersona(personaJeli.Value);
134 
135  StartCoroutine(UpdateBotEnergyAsync());
136  StartCoroutine(UpdateBotTurboAsync());
137  }
138 
139  UJeli traitsDetail = details["Traits"];
140 
141  if (traitsDetail != null)
142  {
143  foreach (var trait in Traits)
144  {
145  UJeli traitDetail = traitsDetail[trait.ID];
146 
147  if (traitDetail != null)
148  trait.Points = traitDetail.IntValue;
149  }
150  }
151  }
152  }
153 
154  public CharacterTrait GetTrait (string traitId)
155  {
156  return Traits.FirstOrDefault(t => t.ID == traitId);
157  }
158 
159  public Persona GetPersona (string personaId)
160  {
161  return Personas.FirstOrDefault(p => p.ID == personaId);
162  }
163 
164  public Persona GetPreviousPersona ()
165  {
166  return Personas[(_activePersonaIndex.LastValue + Personas.Length - 1) % Personas.Length];
167  }
168 
169  public Persona GetNextPersona ()
170  {
171  return Personas[(_activePersonaIndex.LastValue + 1) % Personas.Length];
172  }
173 
174  public void CyclePreviousPersona ()
175  {
176  SetActivePersona(GetPreviousPersona());
177  }
178 
179  public void CycleNextPersona ()
180  {
181  SetActivePersona(GetNextPersona());
182  }
183 
184  public void SetActivePersona (Persona persona)
185  {
186  SetActivePersona(Array.IndexOf(Personas, persona));
187  }
188 
189  public void SetActivePersona (string personaId)
190  {
191  SetActivePersona(Array.FindIndex(Personas, p => p.ID == personaId));
192  }
193 
194  private void SetActivePersona (int personaIndex)
195  {
196  if (!Sync.IsLocal(this))
197  return;
198 
199  _energy = 0f;
200 
201  if (personaIndex != -1)
202  {
203  // TODO: Log persona change.
204  _activePersonaIndex[0] = personaIndex;
205  }
206  else _activePersonaIndex[0] = 0;
207  }
208 
209  public void UseSkill ()
210  {
211  if (!this.IsSkillReady)
212  return;
213 
214  if (Sync.IsClient)
215  Study.LogEvent(SkillLogEvent.UsePrimary, ActivePersona.ID);
216 
217  _skillUses[0] = true;
218  _energy = 0f;
219  }
220 
221  void OnActivePersonaIndexMessage (Timeline<int> timeline, TimelineEntry<int> entry)
222  {
223  Sync.SendEnabledMessage(this, "OnActivePersonaChanged", this.ActivePersona);
224  }
225 
226  void OnTurboMessage (Timeline<bool> timeline, TimelineEntry<bool> entry)
227  {
228  if (entry.Value)
229  Sync.SendEnabledMessage(this, "OnGotTurbo");
230  else Sync.SendEnabledMessage(this, "OnLostTurbo");
231  }
232 
233  void OnSkillUseMessage (Timeline<bool> timeline, TimelineEntry<bool> entry)
234  {
235  Sync.SendEnabledMessage(this, "OnSkillUsed", this, this.ActivePersona);
236  }
237 
238  private IEnumerator UpdateHumanPersonaAsync ()
239  {
240  while (true)
241  {
242  if (CanUseSkill)
243  {
244  var activePersona = this.ActivePersona;
245 
246  if (activePersona != null)
247  {
248  if (activePersona.AutoFireSkill)
249  {
250  if (Controls.GetAction(ControlsAction.Primary))
251  UseSkill();
252  }
253  else
254  {
255  if (Controls.GetActionDown(ControlsAction.Primary))
256  UseSkill();
257  }
258  }
259  }
260 
261  if (_activePersonaIndex.LastValueDuration >= _personaChangeCooldown)
262  {
263  if (Controls.GetActionDown(ControlsAction.CycleNext))
264  CycleNextPersona();
265  else if (Controls.GetActionDown(ControlsAction.CyclePrevious))
266  CyclePreviousPersona();
267  }
268 
269  yield return null;
270  }
271  }
272 
273  private IEnumerator UpdateHumanEnergyAsync ()
274  {
275  while (true)
276  {
277  if (CanUseSkill)
278  {
279  Persona activePersona = this.ActivePersona;
280 
281  if (activePersona != null)
282  {
283  if (Controls.Power >= .1f && CanChargeEnergy)
284  {
285  float prevEnergy = _energy;
286  _energy = Mathf.Min(1, _energy + Controls.Power / activePersona.SkillChargeTime * Time.deltaTime);
287 
288  if (_energy == 1 && prevEnergy != 1)
289  this.PlayPooledSound("get_action");
290  }
291  else _energy = Mathf.Max(0, _energy - 1 / activePersona.SkillChargeTime * Time.deltaTime);
292  }
293  else _energy = 0f;
294  }
295  else _energy = 0f;
296 
297  yield return null;
298  }
299  }
300 
301  private IEnumerator UpdateBotEnergyAsync ()
302  {
303  while (true)
304  {
305  if (CanUseSkill)
306  {
307  Persona activePersona = this.ActivePersona;
308 
309  if (activePersona != null)
310  {
311  _energy += 1 / activePersona.SkillChargeTime * Time.deltaTime;
312  }
313  else _energy = 0f;
314  }
315  else _energy = 0f;
316 
317  yield return null;
318  }
319  }
320 
321  private IEnumerator UpdateHumanTurboAsync ()
322  {
323  while (true)
324  {
325  if (_isTurbo.LastValue)
326  {
327  if (Heart.Tier != HeartTier.Target)
328  {
329  if (_isTurbo.LastValue)
330  {
331  _isTurbo[0] = false;
332  }
333  }
334  }
335  else
336  {
337  if (Heart.Tier == HeartTier.Target)
338  {
339  if (!_isTurbo.LastValue)
340  {
341  _isTurbo[0] = true;
342  }
343  }
344  }
345 
346  yield return null;
347  }
348  }
349 
350  private IEnumerator UpdateBotTurboAsync ()
351  {
352  yield return new WaitForSeconds(Mathf.Max(0f, BotTurboWarmup + UnityEngine.Random.Range(-15f, 15f)));
353  _isTurbo[0] = true;
354  }
355 }
static float Power
Returns an abstract power value in the range [0 ... 1].
Definition: Controls.cs:47
A visual representation of HeartRate and Character.Energy.
static bool IsClient
Gets whether or not this peer is a client.
Definition: Sync.Static.cs:61
static bool IsClientOwned(GameObject go)
Gets whether or not the given object is owned by a client.
Definition: Sync.Static.cs:359
static bool GetAction(ControlsAction action)
Returns the current state of an action.
Definition: Controls.cs:186
string ID
String ID of the persona.
Definition: Persona.cs:24
Heart class. Allows for heart rate queries and heart rate injection from devices. ...
Definition: Heart.cs:20
static void RegisterEnergyProvider(Character provider)
Register an instance of a Character (usually the player's local avatar's Character component) to act ...
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
static bool IsLocal(GameObject go)
Gets whether or not the given object is owned by the local peer.
Definition: Sync.Static.cs:290
static bool GetActionDown(ControlsAction action)
Checks if the given action has just been pressed down.
Definition: Controls.cs:194
(Deprecated) Descibes a trait for a character.
static int GetOwner(GameObject go)
Gets the owner peer index of a given object.
Definition: Sync.Static.cs:263
Avatar character management component. In charge of personas, skills and buffs.
Definition: Character.cs:12
Class for managing study-related game components.
Definition: Study.cs:13
static HeartTier Tier
Gets the current heart tier.
Definition: Heart.cs:49
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
Controls class. Allows for controls queries and controls injection from devices.
Definition: Controls.cs:41
Descibes a persona for an avatar character.
Definition: Persona.cs:9