Liberi
An exergame built for kids with CP!
UTimelineUtils.cs
1 using UnityEngine;
2 using Janus;
3 using System;
4 using System.IO;
5 
9 public static class UTimelineUtils
10 {
14  public static float MaxExtrapolationTimeJump = 1f;
15 
20  public static void SetDefautTimelineFunctions ()
21  {
22  Timeline<Vector2>.TypeEncode = EncodeVector2;
23  Timeline<Vector2>.TypeDecode = DecodeVector2;
24  Timeline<Vector3>.TypeEncode = EncodeVector3;
25  Timeline<Vector3>.TypeDecode = DecodeVector3;
26  Timeline<Vector4>.TypeEncode = EncodeVector4;
27  Timeline<Vector4>.TypeDecode = DecodeVector4;
28  Timeline<Quaternion>.TypeEncode = EncodeQuaternion;
29  Timeline<Quaternion>.TypeDecode = DecodeQuaternion;
30  Timeline<Matrix4x4>.TypeEncode = EncodeMatrix4x4;
31  Timeline<Matrix4x4>.TypeDecode = DecodeMatrix4x4;
32  Timeline<Ray>.TypeEncode = EncodeRay;
33  Timeline<Ray>.TypeDecode = DecodeRay;
34  Timeline<Color>.TypeEncode = EncodeColor;
35  Timeline<Color>.TypeDecode = DecodeColor;
36  Timeline<GameObject>.TypeEncode = EncodeGameObject;
37  Timeline<GameObject>.TypeDecode = DecodeGameObject;
38  Timeline<UJeli>.TypeEncode = EncodeUJeli;
39  Timeline<UJeli>.TypeDecode = DecodeUJeli;
40  Timeline<GameMessage>.TypeEncode = EncodeGameMessage;
41  Timeline<GameMessage>.TypeDecode = DecodeGameMessage;
42 
43  Timeline<Vector2>.TypeInterpolate = TimelineUtils.BuildLinearInterpolator<Vector2>(
44  (x, y) => x + y, (x, y) => x * y);
45  Timeline<Vector2>.TypeExtrapolate = TimelineUtils.BuildLinearExtrapolator<Vector2>(
46  (x, y) => x + y, (x, y) => x * y, MaxExtrapolationTimeJump);
47  Timeline<Vector3>.TypeInterpolate = TimelineUtils.BuildLinearInterpolator<Vector3>(
48  (x, y) => x + y, (x, y) => x * y);
49  Timeline<Vector3>.TypeExtrapolate = TimelineUtils.BuildLinearExtrapolator<Vector3>(
50  (x, y) => x + y, (x, y) => x * y, MaxExtrapolationTimeJump);
51  Timeline<Vector4>.TypeInterpolate = TimelineUtils.BuildLinearInterpolator<Vector4>(
52  (x, y) => x + y, (x, y) => x * y);
53  Timeline<Vector4>.TypeExtrapolate = TimelineUtils.BuildLinearExtrapolator<Vector4>(
54  (x, y) => x + y, (x, y) => x * y, MaxExtrapolationTimeJump);
55 
56  Timeline<Quaternion>.TypeInterpolate = InterpolateQuaternionSlerp;
57  Timeline<Quaternion>.TypeExtrapolate = ExtrapolateQuaternionSlerp;
58  Timeline<Ray>.TypeInterpolate = InterpolateRay;
59  Timeline<Ray>.TypeExtrapolate = ExtrapolateRay;
60  }
61 
62  public static byte[] EncodeVector2 (Vector2 value)
63  {
64  byte[] bytes = new byte[2 * sizeof(float)];
65 
66  BinaryWriter bw = new BinaryWriter(new MemoryStream(bytes));
67  bw.Write(value.x); bw.Write(value.y);
68  bw.Close();
69 
70  return bytes;
71  }
72 
73  public static Vector2 DecodeVector2 (byte[] bytes)
74  {
75  BinaryReader br = new BinaryReader(new MemoryStream(bytes));
76  var value = new Vector2(br.ReadSingle(), br.ReadSingle());
77  br.Close();
78 
79  return value;
80  }
81 
82  public static byte[] EncodeVector3 (Vector3 value)
83  {
84  byte[] bytes = new byte[3 * sizeof(float)];
85 
86  BinaryWriter bw = new BinaryWriter(new MemoryStream(bytes));
87  bw.Write(value.x); bw.Write(value.y); bw.Write(value.z);
88  bw.Close();
89 
90  return bytes;
91  }
92 
93  public static Vector3 DecodeVector3 (byte[] bytes)
94  {
95  BinaryReader br = new BinaryReader(new MemoryStream(bytes));
96  var value = new Vector3(br.ReadSingle(), br.ReadSingle(), br.ReadSingle());
97  br.Close();
98 
99  return value;
100  }
101 
102  public static byte[] EncodeVector4 (Vector4 value)
103  {
104  byte[] bytes = new byte[4 * sizeof(float)];
105 
106  BinaryWriter bw = new BinaryWriter(new MemoryStream(bytes));
107  bw.Write(value.x); bw.Write(value.y); bw.Write(value.z); bw.Write(value.w);
108  bw.Close();
109 
110  return bytes;
111  }
112 
113  public static Vector4 DecodeVector4 (byte[] bytes)
114  {
115  BinaryReader br = new BinaryReader(new MemoryStream(bytes));
116  var value = new Vector4(br.ReadSingle(), br.ReadSingle(), br.ReadSingle(), br.ReadSingle());
117  br.Close();
118 
119  return value;
120  }
121 
122  public static byte[] EncodeQuaternion (Quaternion value)
123  {
124  byte[] bytes = new byte[4 * sizeof(float)];
125 
126  BinaryWriter bw = new BinaryWriter(new MemoryStream(bytes));
127  bw.Write(value.x); bw.Write(value.y); bw.Write(value.z); bw.Write(value.w);
128  bw.Close();
129 
130  return bytes;
131  }
132 
133  public static Quaternion DecodeQuaternion (byte[] bytes)
134  {
135  BinaryReader br = new BinaryReader(new MemoryStream(bytes));
136  var value = new Quaternion(br.ReadSingle(), br.ReadSingle(), br.ReadSingle(), br.ReadSingle());
137  br.Close();
138 
139  return value;
140  }
141 
142  public static byte[] EncodeByteEnum<T> (T value)
143  {
144  return new byte[] { (byte)(object)value };
145  }
146 
147  public static T DecodeByteEnum<T> (byte[] bytes)
148  {
149  return (T)(object)bytes[0];
150  }
151 
152  public static byte[] EncodeIntEnum<T> (T value)
153  {
154  return BitConverter.GetBytes((int)(object)value);
155  }
156 
157  public static T DecodeIntEnum<T> (byte[] bytes)
158  {
159  return (T)(object)BitConverter.ToInt32(bytes, 0);
160  }
161 
162  public static byte[] EncodeShortEnum<T> (T value)
163  {
164  return BitConverter.GetBytes((short)(object)value);
165  }
166 
167  public static T DecodeShortEnum<T> (byte[] bytes)
168  {
169  return (T)(object)BitConverter.ToInt16(bytes, 0);
170  }
171 
172  public static byte[] EncodeEnumByName<T> (T value)
173  {
174  return TimelineUtils.EncodeString(value.ToString());
175  }
176 
177  public static T DecodeEnumByName<T> (byte[] bytes)
178  {
179  return (T)Enum.Parse(typeof(T), TimelineUtils.DecodeString(bytes));
180  }
181 
182  public static byte[] EncodeGameObject (GameObject value)
183  {
184  return BitConverter.GetBytes(Sync.GetObjectIndex(value));
185  }
186 
187  public static GameObject DecodeGameObject (byte[] bytes)
188  {
189  return Sync.GetObject(BitConverter.ToInt32(bytes, 0));
190  }
191 
192  public static byte[] EncodeUJeli (UJeli value)
193  {
194  return TimelineUtils.EncodeString(value.ToString());
195  }
196 
197  public static UJeli DecodeUJeli (byte[] bytes)
198  {
199  return UJeli.Parse(TimelineUtils.DecodeString(bytes));
200  }
201 
202  public static byte[] EncodeGameMessage (GameMessage value)
203  {
204  return value.Data;
205  }
206 
207  public static GameMessage DecodeGameMessage (byte[] bytes)
208  {
209  var msg = new GameMessage();
210  msg.Write(bytes);
211  msg.Position = 0;
212 
213  return msg;
214  }
215 
216  public static Vector2 InterpolateVector2Slerp (Timeline<Vector2> timeline, TimelineContext<Vector2> context)
217  {
218  return Vector3.Slerp(context.Prev.Value, context.Next.Value,
219  (float)(context.Time - context.Prev.Time) / (float)(context.Next.Time - context.Prev.Time));
220  }
221 
222  public static Vector2 ExtrapolateVector2Slerp (Timeline<Vector2> timeline, TimelineContext<Vector3> context)
223  {
224  if (context.Prev.Prev != null && context.Prev.Prev.Time != context.Prev.Time)
225  {
226  return ExtrapolateVector3Slerp(context.Prev.Prev.Value, context.Prev.Value,
227  (float)(context.Prev.Time - context.Prev.Prev.Time), (float)(context.Time - context.Prev.Prev.Time));
228  }
229  else return context.Prev.Value;
230  }
231 
232  public static Vector3 InterpolateVector3Slerp (Timeline<Vector3> timeline, TimelineContext<Vector3> context)
233  {
234  return Vector3.Slerp(context.Prev.Value, context.Next.Value,
235  (float)(context.Time - context.Prev.Time) / (float)(context.Next.Time - context.Prev.Time));
236  }
237 
238  public static Vector3 ExtrapolateVector3Slerp (Timeline<Vector3> timeline, TimelineContext<Vector3> context)
239  {
240  if (context.Prev.Prev != null && context.Prev.Prev.Time != context.Prev.Time)
241  {
242  return ExtrapolateVector3Slerp(context.Prev.Prev.Value, context.Prev.Value,
243  (float)(context.Prev.Time - context.Prev.Prev.Time), (float)(context.Time - context.Prev.Prev.Time));
244  }
245  else return context.Prev.Value;
246  }
247 
248  public static Vector3 ExtrapolateVector3Slerp (Vector3 v1, Vector3 v2, float t12, float t13)
249  {
250  if (v1 == v2)
251  return v1;
252 
253  float timeJump = Mathf.Min(MaxExtrapolationTimeJump, t13 - t12);
254  t13 = t12 + timeJump;
255 
256  var baseRot = Quaternion.FromToRotation(v1, v2);
257 
258  Vector3 rotAxis;
259  float baseRotAngle;
260  baseRot.ToAngleAxis(out baseRotAngle, out rotAxis);
261 
262  float rotSpeed = baseRotAngle / t12;
263  float newRotAngle = rotSpeed * t13;
264 
265  var newRot = Quaternion.AngleAxis(newRotAngle, rotAxis);
266  return newRot * v1;
267  }
268 
269  public static Quaternion InterpolateQuaternionSlerp (Timeline<Quaternion> timeline,
270  TimelineContext<Quaternion> context)
271  {
272  return Quaternion.Slerp(context.Prev.Value, context.Next.Value,
273  (float)(context.Time - context.Prev.Time) / (float)(context.Next.Time - context.Prev.Time));
274  }
275 
276  public static Quaternion ExtrapolateQuaternionSlerp (Timeline<Quaternion> timeline,
277  TimelineContext<Quaternion> context)
278  {
279  if (context.Prev.Prev != null && context.Prev.Prev.Time != context.Prev.Time)
280  {
281  return ExtrapolateQuaternionSlerp(
282  context.Prev.Prev.Value, context.Prev.Value,
283  (float)(context.Prev.Time - context.Prev.Prev.Time),
284  (float)(context.Time - context.Prev.Prev.Time));
285  }
286  else return context.Prev.Value;
287  }
288 
289  public static Quaternion ExtrapolateQuaternionSlerp (Quaternion q1, Quaternion q2, float t12, float t13)
290  {
291  if (q1 == q2)
292  return q1;
293 
294  float timeJump = Mathf.Min(MaxExtrapolationTimeJump, t13 - t12);
295  t13 = t12 + timeJump;
296 
297  var baseRot = q2 * Quaternion.Inverse(q1);
298 
299  Vector3 rotAxis;
300  float baseRotAngle;
301  baseRot.ToAngleAxis(out baseRotAngle, out rotAxis);
302 
303  float rotSpeed = baseRotAngle / t12;
304  float newRotAngle = rotSpeed * t13;
305 
306  var newRot = Quaternion.AngleAxis(newRotAngle, rotAxis);
307  return newRot * q1;
308  }
309 
310  public static float InterpolateAngleSlerp (Timeline<float> timeline, TimelineContext<float> context)
311  {
312  return Mathf.LerpAngle(context.Prev.Value, context.Next.Value,
313  (float)(context.Time - context.Prev.Time) / (float)(context.Next.Time - context.Prev.Time));
314  }
315 
316  public static float ExtrapolateAngleSlerp (Timeline<float> timeline, TimelineContext<float> context)
317  {
318  if (context.Prev.Prev != null && context.Prev.Prev.Time != context.Prev.Time)
319  {
320  return ExtrapolateAngleSlerp(
321  context.Prev.Prev.Value, context.Prev.Value,
322  (float)(context.Prev.Time - context.Prev.Prev.Time),
323  (float)(context.Time - context.Prev.Prev.Time));
324  }
325  else return context.Prev.Value;
326  }
327 
328  public static float ExtrapolateAngleSlerp (float a1, float a2, float t12, float t13)
329  {
330  if (a1 == a2)
331  return a1;
332 
333  float timeJump = Mathf.Min(MaxExtrapolationTimeJump, t13 - t12);
334  t13 = t12 + timeJump;
335 
336  return NormalizeAngle(ExtrapolateQuaternionSlerp(
337  Quaternion.Euler(0, 0, NormalizeAngle(a1)),
338  Quaternion.Euler(0, 0, NormalizeAngle(a2)), t12, t13).eulerAngles.z);
339  }
340 
341  public static float NormalizeAngle (float a)
342  {
343  while (a > 180) { a -= 360; }
344  while (a < -180) { a += 360; }
345 
346  return a;
347  }
348 
349  public static byte[] EncodeMatrix4x4 (Matrix4x4 value)
350  {
351  byte[] bytes = new byte[16 * sizeof(float)];
352 
353  BinaryWriter bw = new BinaryWriter(new MemoryStream(bytes));
354  bw.Write(value.m00); bw.Write(value.m01); bw.Write(value.m02); bw.Write(value.m03);
355  bw.Write(value.m10); bw.Write(value.m11); bw.Write(value.m12); bw.Write(value.m13);
356  bw.Write(value.m20); bw.Write(value.m21); bw.Write(value.m22); bw.Write(value.m23);
357  bw.Write(value.m30); bw.Write(value.m31); bw.Write(value.m32); bw.Write(value.m33);
358  bw.Close();
359 
360  return bytes;
361  }
362 
363  public static Matrix4x4 DecodeMatrix4x4 (byte[] bytes)
364  {
365  var value = new Matrix4x4();
366 
367  BinaryReader br = new BinaryReader(new MemoryStream(bytes));
368  value.SetRow(0, new Vector4(br.ReadSingle(), br.ReadSingle(), br.ReadSingle(), br.ReadSingle()));
369  value.SetRow(1, new Vector4(br.ReadSingle(), br.ReadSingle(), br.ReadSingle(), br.ReadSingle()));
370  value.SetRow(2, new Vector4(br.ReadSingle(), br.ReadSingle(), br.ReadSingle(), br.ReadSingle()));
371  value.SetRow(3, new Vector4(br.ReadSingle(), br.ReadSingle(), br.ReadSingle(), br.ReadSingle()));
372  br.Close();
373 
374  return value;
375  }
376 
377  public static byte[] EncodeRay (Ray value)
378  {
379  byte[] bytes = new byte[6 * sizeof(float)];
380 
381  BinaryWriter bw = new BinaryWriter(new MemoryStream(bytes));
382  bw.Write(value.origin.x); bw.Write(value.origin.y); bw.Write(value.origin.z);
383  bw.Write(value.direction.x); bw.Write(value.direction.y); bw.Write(value.direction.z);
384  bw.Close();
385 
386  return bytes;
387  }
388 
389  public static Ray DecodeRay (byte[] bytes)
390  {
391  BinaryReader br = new BinaryReader(new MemoryStream(bytes));
392  var value = new Ray(
393  new Vector3(br.ReadSingle(), br.ReadSingle(), br.ReadSingle()),
394  new Vector3(br.ReadSingle(), br.ReadSingle(), br.ReadSingle()));
395  br.Close();
396 
397  return value;
398  }
399 
400  public static Ray InterpolateRay (Timeline<Ray> timeline, TimelineContext<Ray> context)
401  {
402  float interpFactor = (float)(context.Time - context.Prev.Time) / (float)(context.Next.Time - context.Prev.Time);
403 
404  return new Ray(
405  Vector3.Lerp(context.Prev.Value.origin, context.Next.Value.origin, interpFactor),
406  Vector3.Slerp(context.Prev.Value.direction, context.Next.Value.direction, interpFactor));
407  }
408 
409  public static Ray ExtrapolateRay (Timeline<Ray> timeline, TimelineContext<Ray> context)
410  {
411  if (context.Prev.Prev != null && context.Prev.Prev.Time != context.Prev.Time)
412  {
413  return context.Prev.Value;
414  }
415  else return context.Prev.Value;
416  }
417 
418  public static byte[] EncodeColor (Color value)
419  {
420  byte[] bytes = new byte[4 * sizeof(float)];
421 
422  BinaryWriter bw = new BinaryWriter(new MemoryStream(bytes));
423  bw.Write(value.r); bw.Write(value.g); bw.Write(value.b); bw.Write(value.a);
424  bw.Close();
425 
426  return bytes;
427  }
428 
429  public static Color DecodeColor (byte[] bytes)
430  {
431  BinaryReader br = new BinaryReader(new MemoryStream(bytes));
432  var value = new Color(br.ReadSingle(), br.ReadSingle(), br.ReadSingle(), br.ReadSingle());
433  br.Close();
434 
435  return value;
436  }
437 
438  public static bool HasPart<T> (this TimelineEntry<T> entry, T part)
439  {
440  int valueInt = (int)Convert.ChangeType(entry.Value, typeof(int));
441  int partInt = (int)Convert.ChangeType(part, typeof(int));
442 
443  return (valueInt & partInt) != 0;
444  }
445 
446  public static bool GotPart<T> (this TimelineEntry<T> entry, T part)
447  {
448  int valueInt = (int)Convert.ChangeType(entry.Value, typeof(int));
449  int partInt = (int)Convert.ChangeType(part, typeof(int));
450 
451  return (valueInt & partInt) == partInt &&
452  (entry.Prev == null || ((int)Convert.ChangeType(entry.Prev.Value, typeof(int)) & partInt) == 0);
453  }
454 
455  public static bool LostPart<T> (this TimelineEntry<T> entry, T part)
456  {
457  int valueInt = (int)Convert.ChangeType(entry.Value, typeof(int));
458  int partInt = (int)Convert.ChangeType(part, typeof(int));
459 
460  return (valueInt & partInt) == 0 &&
461  (entry.Prev != null && ((int)Convert.ChangeType(entry.Prev.Value, typeof(int)) & partInt) == partInt);
462  }
463 }
static int GetObjectIndex(GameObject go)
Gets the object index of a given object.
Definition: Sync.Static.cs:190
A class for serializing game data into a stream for propagation between objects and peers...
Definition: GameMessage.cs:14
static GameObject GetObject(int objectIndex)
Get an object by its object index.
Definition: Sync.Static.cs:143
Unity version of Jeli markup class.
Definition: UJeli.cs:10
This class server two main functions: 1) As a MonoBehaviour, it allows for network synchronization of...
Definition: Sync.cs:13