1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
|
using UnityEngine;
using System.Collections;
using System.Collections.Generic;
using Pathfinding.WindowsStore;
using Pathfinding.Serialization;
using Pathfinding.Util;
#if UNITY_WINRT && !UNITY_EDITOR
//using MarkerMetro.Unity.WinLegacy.IO;
//using MarkerMetro.Unity.WinLegacy.Reflection;
#endif
namespace Pathfinding {
[System.Serializable]
/// <summary>
/// Stores the navigation graphs for the A* Pathfinding System.
///
/// An instance of this class is assigned to <see cref="AstarPath.data"/>. From it you can access all graphs loaded through the <see cref="graphs"/> variable.
/// This class also handles a lot of the high level serialization.
/// </summary>
public class AstarData {
/// <summary>The AstarPath component which owns this AstarData</summary>
internal AstarPath active;
#region Fields
/// <summary>Shortcut to the first <see cref="NavMeshGraph"/></summary>
public NavMeshGraph navmesh { get; private set; }
#if !ASTAR_NO_GRID_GRAPH
/// <summary>Shortcut to the first <see cref="GridGraph"/></summary>
public GridGraph gridGraph { get; private set; }
/// <summary>Shortcut to the first <see cref="LayerGridGraph"/>.</summary>
public LayerGridGraph layerGridGraph { get; private set; }
#endif
#if !ASTAR_NO_POINT_GRAPH
/// <summary>Shortcut to the first <see cref="PointGraph"/>.</summary>
public PointGraph pointGraph { get; private set; }
#endif
/// <summary>Shortcut to the first <see cref="RecastGraph"/>.</summary>
public RecastGraph recastGraph { get; private set; }
/// <summary>Shortcut to the first <see cref="LinkGraph"/>.</summary>
public LinkGraph linkGraph { get; private set; }
/// <summary>
/// All supported graph types.
/// Populated through reflection search
/// </summary>
public System.Type[] graphTypes { get; private set; }
#if ASTAR_FAST_NO_EXCEPTIONS || UNITY_WINRT || UNITY_WEBGL
/// <summary>
/// Graph types to use when building with Fast But No Exceptions for iPhone.
/// If you add any custom graph types, you need to add them to this hard-coded list.
/// </summary>
public static readonly System.Type[] DefaultGraphTypes = new System.Type[] {
#if !ASTAR_NO_GRID_GRAPH
typeof(GridGraph),
typeof(LayerGridGraph),
#endif
#if !ASTAR_NO_POINT_GRAPH
typeof(PointGraph),
#endif
typeof(NavMeshGraph),
typeof(RecastGraph),
typeof(LinkGraph),
};
#endif
/// <summary>
/// All graphs.
/// This will be filled only after deserialization has completed.
/// May contain null entries if graph have been removed.
/// </summary>
[System.NonSerialized]
public NavGraph[] graphs = new NavGraph[0];
/// <summary>
/// Serialized data for all graphs and settings.
/// Stored as a base64 encoded string because otherwise Unity's Undo system would sometimes corrupt the byte data (because it only stores deltas).
///
/// This can be accessed as a byte array from the <see cref="data"/> property.
/// </summary>
[SerializeField]
string dataString;
/// <summary>Serialized data for all graphs and settings</summary>
private byte[] data {
get {
var d = dataString != null? System.Convert.FromBase64String(dataString) : null;
// Unity can initialize the dataString to an empty string, but that's not a valid zip file
if (d != null && d.Length == 0) return null;
return d;
}
set {
dataString = value != null? System.Convert.ToBase64String(value) : null;
}
}
/// <summary>
/// Serialized data for cached startup.
/// If set, on start the graphs will be deserialized from this file.
/// </summary>
public TextAsset file_cachedStartup;
/// <summary>
/// Should graph-data be cached.
/// Caching the startup means saving the whole graphs - not only the settings - to a file (<see cref="file_cachedStartup)"/> which can
/// be loaded when the game starts. This is usually much faster than scanning the graphs when the game starts. This is configured from the editor under the "Save & Load" tab.
///
/// See: save-load-graphs (view in online documentation for working links)
/// </summary>
[SerializeField]
public bool cacheStartup;
//End Serialization Settings
List<bool> graphStructureLocked = new List<bool>();
#endregion
internal AstarData (AstarPath active) {
this.active = active;
}
public byte[] GetData() => data;
public void SetData (byte[] data) {
this.data = data;
}
/// <summary>Loads the graphs from memory, will load cached graphs if any exists</summary>
public void OnEnable () {
if (graphTypes == null) {
FindGraphTypes();
}
if (graphs == null) graphs = new NavGraph[0];
if (cacheStartup && file_cachedStartup != null && Application.isPlaying) {
LoadFromCache();
} else {
DeserializeGraphs();
}
}
/// <summary>
/// Prevent the graph structure from changing during the time this lock is held.
/// This prevents graphs from being added or removed and also prevents graphs from being serialized or deserialized.
/// This is used when e.g an async scan is happening to ensure that for example a graph that is being scanned is not destroyed.
///
/// Each call to this method *must* be paired with exactly one call to <see cref="UnlockGraphStructure"/>.
/// The calls may be nested.
/// </summary>
internal void LockGraphStructure (bool allowAddingGraphs = false) {
graphStructureLocked.Add(allowAddingGraphs);
}
/// <summary>
/// Allows the graph structure to change again.
/// See: <see cref="LockGraphStructure"/>
/// </summary>
internal void UnlockGraphStructure () {
if (graphStructureLocked.Count == 0) throw new System.InvalidOperationException();
graphStructureLocked.RemoveAt(graphStructureLocked.Count - 1);
}
PathProcessor.GraphUpdateLock AssertSafe (bool onlyAddingGraph = false) {
if (graphStructureLocked.Count > 0) {
bool allowAdding = true;
for (int i = 0; i < graphStructureLocked.Count; i++) allowAdding &= graphStructureLocked[i];
if (!(onlyAddingGraph && allowAdding)) throw new System.InvalidOperationException("Graphs cannot be added, removed or serialized while the graph structure is locked. This is the case when a graph is currently being scanned and when executing graph updates and work items.\nHowever as a special case, graphs can be added inside work items.");
}
// Pause the pathfinding threads
var graphLock = active.PausePathfinding();
if (!active.IsInsideWorkItem) {
// Make sure all graph updates and other callbacks are done
// Only do this if this code is not being called from a work item itself as that would cause a recursive wait that could never complete.
// There are some valid cases when this can happen. For example it may be necessary to add a new graph inside a work item.
active.FlushWorkItems();
// Paths that are already calculated and waiting to be returned to the Seeker component need to be
// processed immediately as their results usually depend on graphs that currently exist. If this was
// not done then after destroying a graph one could get a path result with destroyed nodes in it.
active.pathReturnQueue.ReturnPaths(false);
}
return graphLock;
}
/// <summary>
/// Calls the callback with every node in all graphs.
/// This is the easiest way to iterate through every existing node.
///
/// <code>
/// AstarPath.active.data.GetNodes(node => {
/// Debug.Log("I found a node at position " + (Vector3)node.position);
/// });
/// </code>
///
/// See: <see cref="Pathfinding.NavGraph.GetNodes"/> for getting the nodes of a single graph instead of all.
/// See: graph-updates (view in online documentation for working links)
/// </summary>
public void GetNodes (System.Action<GraphNode> callback) {
for (int i = 0; i < graphs.Length; i++) {
if (graphs[i] != null) graphs[i].GetNodes(callback);
}
}
/// <summary>
/// Updates shortcuts to the first graph of different types.
/// Hard coding references to some graph types is not really a good thing imo. I want to keep it dynamic and flexible.
/// But these references ease the use of the system, so I decided to keep them.
/// </summary>
public void UpdateShortcuts () {
navmesh = (NavMeshGraph)FindGraphOfType(typeof(NavMeshGraph));
#if !ASTAR_NO_GRID_GRAPH
gridGraph = (GridGraph)FindGraphOfType(typeof(GridGraph));
layerGridGraph = (LayerGridGraph)FindGraphOfType(typeof(LayerGridGraph));
#endif
#if !ASTAR_NO_POINT_GRAPH
pointGraph = (PointGraph)FindGraphOfType(typeof(PointGraph));
#endif
recastGraph = (RecastGraph)FindGraphOfType(typeof(RecastGraph));
linkGraph = (LinkGraph)FindGraphOfType(typeof(LinkGraph));
}
/// <summary>Load from data from <see cref="file_cachedStartup"/></summary>
public void LoadFromCache () {
var graphLock = AssertSafe();
if (file_cachedStartup != null) {
var bytes = file_cachedStartup.bytes;
DeserializeGraphs(bytes);
GraphModifier.TriggerEvent(GraphModifier.EventType.PostCacheLoad);
} else {
Debug.LogError("Can't load from cache since the cache is empty");
}
graphLock.Release();
}
#region Serialization
/// <summary>
/// Serializes all graphs settings to a byte array.
/// See: DeserializeGraphs(byte[])
/// </summary>
public byte[] SerializeGraphs () {
return SerializeGraphs(SerializeSettings.Settings);
}
/// <summary>
/// Serializes all graphs settings and optionally node data to a byte array.
/// See: DeserializeGraphs(byte[])
/// See: Pathfinding.Serialization.SerializeSettings
/// </summary>
public byte[] SerializeGraphs (SerializeSettings settings) {
return SerializeGraphs(settings, out var _);
}
/// <summary>
/// Main serializer function.
/// Serializes all graphs to a byte array
/// A similar function exists in the AstarPathEditor.cs script to save additional info
/// </summary>
public byte[] SerializeGraphs (SerializeSettings settings, out uint checksum) {
var graphLock = AssertSafe();
var sr = new AstarSerializer(this, settings, active.gameObject);
sr.OpenSerialize();
sr.SerializeGraphs(graphs);
sr.SerializeExtraInfo();
byte[] bytes = sr.CloseSerialize();
checksum = sr.GetChecksum();
#if ASTARDEBUG
Debug.Log("Got a whole bunch of data, "+bytes.Length+" bytes");
#endif
graphLock.Release();
return bytes;
}
/// <summary>Deserializes graphs from <see cref="data"/></summary>
public void DeserializeGraphs () {
var dataBytes = data;
if (dataBytes != null) {
DeserializeGraphs(dataBytes);
}
}
/// <summary>
/// Destroys all graphs and sets <see cref="graphs"/> to null.
/// See: <see cref="RemoveGraph"/>
/// </summary>
public void ClearGraphs () {
var graphLock = AssertSafe();
ClearGraphsInternal();
graphLock.Release();
}
void ClearGraphsInternal () {
if (graphs == null) return;
var graphLock = AssertSafe();
for (int i = 0; i < graphs.Length; i++) {
if (graphs[i] != null) {
active.DirtyBounds(graphs[i].bounds);
((IGraphInternals)graphs[i]).OnDestroy();
graphs[i].active = null;
}
}
graphs = new NavGraph[0];
UpdateShortcuts();
graphLock.Release();
}
public void DisposeUnmanagedData () {
if (graphs == null) return;
var graphLock = AssertSafe();
for (int i = 0; i < graphs.Length; i++) {
if (graphs[i] != null) {
((IGraphInternals)graphs[i]).DisposeUnmanagedData();
}
}
graphLock.Release();
}
/// <summary>Makes all graphs become unscanned</summary>
internal void DestroyAllNodes () {
if (graphs == null) return;
var graphLock = AssertSafe();
for (int i = 0; i < graphs.Length; i++) {
if (graphs[i] != null) {
((IGraphInternals)graphs[i]).DestroyAllNodes();
}
}
graphLock.Release();
}
public void OnDestroy () {
ClearGraphsInternal();
}
/// <summary>
/// Deserializes graphs from the specified byte array.
/// An error will be logged if deserialization fails.
/// </summary>
public void DeserializeGraphs (byte[] bytes) {
var graphLock = AssertSafe();
ClearGraphs();
DeserializeGraphsAdditive(bytes);
graphLock.Release();
}
/// <summary>
/// Deserializes graphs from the specified byte array additively.
/// An error will be logged if deserialization fails.
/// This function will add loaded graphs to the current ones.
/// </summary>
public void DeserializeGraphsAdditive (byte[] bytes) {
var graphLock = AssertSafe();
try {
if (bytes != null) {
var sr = new AstarSerializer(this, active.gameObject);
if (sr.OpenDeserialize(bytes)) {
DeserializeGraphsPartAdditive(sr);
sr.CloseDeserialize();
} else {
Debug.Log("Invalid data file (cannot read zip).\nThe data is either corrupt or it was saved using a 3.0.x or earlier version of the system");
}
} else {
throw new System.ArgumentNullException(nameof(bytes));
}
active.VerifyIntegrity();
} catch (System.Exception e) {
Debug.LogError(new System.Exception("Caught exception while deserializing data.", e));
graphs = new NavGraph[0];
}
UpdateShortcuts();
GraphModifier.TriggerEvent(GraphModifier.EventType.PostGraphLoad);
graphLock.Release();
}
/// <summary>Helper function for deserializing graphs</summary>
void DeserializeGraphsPartAdditive (AstarSerializer sr) {
if (graphs == null) graphs = new NavGraph[0];
var gr = new List<NavGraph>(graphs);
// Set an offset so that the deserializer will load
// the graphs with the correct graph indexes
sr.SetGraphIndexOffset(gr.Count);
FindGraphTypes();
var newGraphs = sr.DeserializeGraphs(graphTypes);
gr.AddRange(newGraphs);
if (gr.Count > GraphNode.MaxGraphIndex + 1) {
throw new System.InvalidOperationException("Graph Count Limit Reached. You cannot have more than " + GraphNode.MaxGraphIndex + " graphs.");
}
graphs = gr.ToArray();
// Assign correct graph indices.
for (int i = 0; i < graphs.Length; i++) {
if (graphs[i] == null) continue;
graphs[i].GetNodes(node => node.GraphIndex = (uint)i);
}
for (int i = 0; i < graphs.Length; i++) {
for (int j = i+1; j < graphs.Length; j++) {
if (graphs[i] != null && graphs[j] != null && graphs[i].guid == graphs[j].guid) {
Debug.LogWarning("Guid Conflict when importing graphs additively. Imported graph will get a new Guid.\nThis message is (relatively) harmless.");
graphs[i].guid = Pathfinding.Util.Guid.NewGuid();
break;
}
}
}
sr.PostDeserialization();
// This will refresh off-mesh links,
// and also recalculate the hierarchical graph if necessary
active.AddWorkItem(ctx => {
for (int i = 0; i < newGraphs.Length; i++) {
ctx.DirtyBounds(newGraphs[i].bounds);
}
});
active.FlushWorkItems();
}
#endregion
/// <summary>
/// Find all graph types supported in this build.
/// Using reflection, the assembly is searched for types which inherit from NavGraph.
/// </summary>
public void FindGraphTypes () {
if (graphTypes != null) return;
#if !ASTAR_FAST_NO_EXCEPTIONS && !UNITY_WINRT && !UNITY_WEBGL
var graphList = new List<System.Type>();
foreach (var assembly in System.AppDomain.CurrentDomain.GetAssemblies()) {
System.Type[] types = null;
try {
types = assembly.GetTypes();
} catch {
// Ignore type load exceptions and things like that.
// We might not be able to read all assemblies for some reason, but hopefully the relevant types exist in the assemblies that we can read
continue;
}
foreach (var type in types) {
#if NETFX_CORE && !UNITY_EDITOR
System.Type baseType = type.GetTypeInfo().BaseType;
#else
var baseType = type.BaseType;
#endif
while (baseType != null) {
if (System.Type.Equals(baseType, typeof(NavGraph))) {
graphList.Add(type);
break;
}
#if NETFX_CORE && !UNITY_EDITOR
baseType = baseType.GetTypeInfo().BaseType;
#else
baseType = baseType.BaseType;
#endif
}
}
}
graphTypes = graphList.ToArray();
#if ASTARDEBUG
Debug.Log("Found "+graphTypes.Length+" graph types");
#endif
#else
graphTypes = DefaultGraphTypes;
#endif
}
#region GraphCreation
/// <summary>Creates a new graph instance of type type</summary>
internal NavGraph CreateGraph (System.Type type) {
var graph = System.Activator.CreateInstance(type) as NavGraph;
graph.active = active;
return graph;
}
/// <summary>
/// Adds a graph of type T to the <see cref="graphs"/> array.
/// See: runtime-graphs (view in online documentation for working links)
/// </summary>
public T AddGraph<T> () where T : NavGraph => AddGraph(typeof(T)) as T;
/// <summary>
/// Adds a graph of type type to the <see cref="graphs"/> array.
/// See: runtime-graphs (view in online documentation for working links)
/// </summary>
public NavGraph AddGraph (System.Type type) {
NavGraph graph = null;
for (int i = 0; i < graphTypes.Length; i++) {
if (System.Type.Equals(graphTypes[i], type)) {
graph = CreateGraph(graphTypes[i]);
}
}
if (graph == null) {
Debug.LogError("No NavGraph of type '"+type+"' could be found, "+graphTypes.Length+" graph types are avaliable");
return null;
}
AddGraph(graph);
return graph;
}
/// <summary>Adds the specified graph to the <see cref="graphs"/> array</summary>
void AddGraph (NavGraph graph) {
// Make sure to not interfere with pathfinding
var graphLock = AssertSafe(true);
// Try to fill in an empty position
bool foundEmpty = false;
for (int i = 0; i < graphs.Length; i++) {
if (graphs[i] == null) {
graphs[i] = graph;
graph.graphIndex = (uint)i;
foundEmpty = true;
break;
}
}
if (!foundEmpty) {
if (graphs != null && graphs.Length >= GraphNode.MaxGraphIndex) {
throw new System.Exception("Graph Count Limit Reached. You cannot have more than " + GraphNode.MaxGraphIndex + " graphs.");
}
// Add a new entry to the list
Memory.Realloc(ref graphs, graphs.Length + 1);
graphs[graphs.Length - 1] = graph;
graph.graphIndex = (uint)(graphs.Length-1);
}
UpdateShortcuts();
graph.active = active;
graphLock.Release();
}
/// <summary>
/// Removes the specified graph from the <see cref="graphs"/> array and Destroys it in a safe manner.
/// To avoid changing graph indices for the other graphs, the graph is simply nulled in the array instead
/// of actually removing it from the array.
/// The empty position will be reused if a new graph is added.
///
/// Returns: True if the graph was sucessfully removed (i.e it did exist in the <see cref="graphs"/> array). False otherwise.
///
/// See: <see cref="ClearGraphs"/>
/// </summary>
public bool RemoveGraph (NavGraph graph) {
// Make sure the pathfinding threads are paused
var graphLock = AssertSafe();
active.DirtyBounds(graph.bounds);
((IGraphInternals)graph).OnDestroy();
graph.active = null;
int i = System.Array.IndexOf(graphs, graph);
if (i != -1) graphs[i] = null;
UpdateShortcuts();
active.offMeshLinks.Refresh();
graphLock.Release();
return i != -1;
}
#endregion
#region GraphUtility
/// <summary>
/// Returns the graph which contains the specified node.
/// The graph must be in the <see cref="graphs"/> array.
///
/// Returns: Returns the graph which contains the node. Null if the graph wasn't found
/// </summary>
public static NavGraph GetGraph (GraphNode node) {
if (node == null || node.Destroyed) return null;
AstarPath script = AstarPath.active;
if (System.Object.ReferenceEquals(script, null)) return null;
AstarData data = script.data;
if (data == null || data.graphs == null) return null;
uint graphIndex = node.GraphIndex;
return data.graphs[(int)graphIndex];
}
/// <summary>Returns the first graph which satisfies the predicate. Returns null if no graph was found.</summary>
public NavGraph FindGraph (System.Func<NavGraph, bool> predicate) {
if (graphs != null) {
for (int i = 0; i < graphs.Length; i++) {
if (graphs[i] != null && predicate(graphs[i])) {
return graphs[i];
}
}
}
return null;
}
/// <summary>Returns the first graph of type type found in the <see cref="graphs"/> array. Returns null if no graph was found.</summary>
public NavGraph FindGraphOfType (System.Type type) {
return FindGraph(graph => System.Type.Equals(graph.GetType(), type));
}
/// <summary>Returns the first graph which inherits from the type type. Returns null if no graph was found.</summary>
public NavGraph FindGraphWhichInheritsFrom (System.Type type) {
return FindGraph(graph => WindowsStoreCompatibility.GetTypeInfo(type).IsAssignableFrom(WindowsStoreCompatibility.GetTypeInfo(graph.GetType())));
}
/// <summary>
/// Loop through this function to get all graphs of type 'type'
/// <code>
/// foreach (GridGraph graph in AstarPath.data.FindGraphsOfType (typeof(GridGraph))) {
/// //Do something with the graph
/// }
/// </code>
/// See: AstarPath.RegisterSafeNodeUpdate
/// </summary>
public IEnumerable FindGraphsOfType (System.Type type) {
if (graphs == null) yield break;
for (int i = 0; i < graphs.Length; i++) {
if (graphs[i] != null && System.Type.Equals(graphs[i].GetType(), type)) {
yield return graphs[i];
}
}
}
/// <summary>
/// All graphs which implements the UpdateableGraph interface
/// <code> foreach (IUpdatableGraph graph in AstarPath.data.GetUpdateableGraphs ()) {
/// //Do something with the graph
/// } </code>
/// See: AstarPath.AddWorkItem
/// See: Pathfinding.IUpdatableGraph
/// </summary>
public IEnumerable GetUpdateableGraphs () {
if (graphs == null) yield break;
for (int i = 0; i < graphs.Length; i++) {
if (graphs[i] is IUpdatableGraph) {
yield return graphs[i];
}
}
}
/// <summary>Gets the index of the NavGraph in the <see cref="graphs"/> array</summary>
public int GetGraphIndex (NavGraph graph) {
if (graph == null) throw new System.ArgumentNullException("graph");
var index = -1;
if (graphs != null) {
index = System.Array.IndexOf(graphs, graph);
if (index == -1) Debug.LogError("Graph doesn't exist");
}
return index;
}
#endregion
}
}
|