-
Notifications
You must be signed in to change notification settings - Fork 18
/
Copy pathnodefilter.go
524 lines (445 loc) · 15.7 KB
/
nodefilter.go
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
package tetra3d
import (
"log"
"math/rand"
"regexp"
"sort"
"sync"
"time"
)
var localRandom = rand.New(rand.NewSource(time.Now().Unix()))
const (
nfSortModeNone = iota
nfSortModeAxisX
nfSortModeAxisY
nfSortModeAxisZ
nfSortModeDistance
nfSortModeRandom
)
// NodeIterator is an interface that allows calling a callback for each node in a collection.
type NodeIterator interface {
ForEach(forEachFunc func(node INode) bool)
}
// NodeCollection represents a collection of Nodes.
type NodeCollection[E INode] []E
// ForEach is a function that allows a NodeCollection to fulfill a NodeIterator interface.
func (n NodeCollection[E]) ForEach(forEachFunc func(node INode) bool) {
for _, node := range n {
if !forEachFunc(node) {
break
}
}
}
func (n *NodeCollection[E]) Add(node E) {
*n = append(*n, node)
}
// NodeFilter represents a chain of node filters, executed in sequence to collect the desired nodes
// out of an entire hierarchy. The filters are executed lazily (so only one slice is allocated
// in the process, and possibly one more for the end result, if you get the result as not just a
// slice of INodes).
type NodeFilter struct {
Filters []func(INode) bool // The slice of filters that are currently active on the NodeFilter.
Start INode // The start (root) of the filter.
stopOnFiltered bool // If the filter should continue through to a node's children if the node itself doesn't pass the filter
MaxDepth int // How deep the node filter should search in the starting node's hierarchy; a value that is less than zero means the entire tree will be traversed.
depth int
sortMode int
reverseSort bool
sortTo Vector3
multithreadingWG *sync.WaitGroup
}
func newNodeFilter(startingNode INode) *NodeFilter {
return &NodeFilter{
Start: startingNode,
depth: -1,
MaxDepth: -1,
multithreadingWG: &sync.WaitGroup{},
}
}
func (nf *NodeFilter) execute(node INode) []INode {
nf.depth++
out := []INode{}
added := true
if node != nf.Start {
add := true
for _, filter := range nf.Filters {
if !filter(node) {
add = false
added = false
}
}
if add {
out = append(out, node)
}
}
if nf.MaxDepth < 0 || nf.depth <= nf.MaxDepth {
if !nf.stopOnFiltered || added {
for _, child := range node.Children() {
out = append(out, nf.execute(child)...)
}
}
}
nf.depth--
// If the depth is at -1, then this should be the end
if nf.depth == -1 && nf.sortMode != nfSortModeNone {
switch nf.sortMode {
case nfSortModeAxisX:
sort.SliceStable(out, func(i, j int) bool {
if nf.reverseSort {
return out[i].WorldPosition().X > out[j].WorldPosition().X
}
return out[i].WorldPosition().X < out[j].WorldPosition().X
})
case nfSortModeAxisY:
sort.SliceStable(out, func(i, j int) bool {
if nf.reverseSort {
return out[i].WorldPosition().Y > out[j].WorldPosition().Y
}
return out[i].WorldPosition().Y < out[j].WorldPosition().Y
})
case nfSortModeAxisZ:
sort.SliceStable(out, func(i, j int) bool {
if nf.reverseSort {
return out[i].WorldPosition().Z > out[j].WorldPosition().Z
}
return out[i].WorldPosition().Z < out[j].WorldPosition().Z
})
case nfSortModeDistance:
sort.SliceStable(out, func(i, j int) bool {
if nf.reverseSort {
return out[i].WorldPosition().DistanceSquared(nf.sortTo) > out[j].WorldPosition().DistanceSquared(nf.sortTo)
}
return out[i].WorldPosition().DistanceSquared(nf.sortTo) < out[j].WorldPosition().DistanceSquared(nf.sortTo)
})
case nfSortModeRandom:
localRandom.Shuffle(len(out), func(i, j int) { out[i], out[j] = out[j], out[i] })
}
nf.sortMode = nfSortModeNone
nf.sortTo = Vector3{}
nf.reverseSort = false
}
return out
}
func (nf *NodeFilter) executeFilters(node INode, execute func(INode) bool, multithreading bool) bool {
// TODO: Append filtered nodes to an internal shared slice so sorting could be available
// while also not allocating memory constantly unnecessarily.
if nf.depth < 0 && nf.sortMode != nfSortModeNone {
log.Printf("Warning: NodeFilter executing on Node < %s > has sorting on it, but ForEach() cannot be used with sorting.\n", nf.Start.Path())
}
nf.depth++
successfulFilter := true
if node != nf.Start {
ok := true
for _, filter := range nf.Filters {
if !filter(node) {
ok = false
successfulFilter = false
}
}
if ok {
if multithreading {
nf.multithreadingWG.Add(1)
go func() {
defer nf.multithreadingWG.Done()
execute(node)
}()
} else {
if !execute(node) {
nf.depth--
return false
}
}
}
}
if nf.MaxDepth < 0 || nf.depth <= nf.MaxDepth {
if !nf.stopOnFiltered || successfulFilter {
includeChild := true
node.ForEachChild(func(child INode, index int, size int) bool {
if !nf.executeFilters(child, execute, multithreading) {
includeChild = false
nf.depth--
return false
}
return true
})
if !includeChild {
return false
}
}
}
nf.depth--
if multithreading && node == nf.Start {
nf.multithreadingWG.Wait()
}
return true
}
// First returns the first Node in the NodeFilter; if the NodeFilter is empty, this function returns nil.
func (nf NodeFilter) First() INode {
var result INode
nf.ForEach(func(node INode) bool { result = node; return false })
return result
}
// First returns the last Node in the NodeFilter; if the NodeFilter is empty, this function returns nil.
func (nf NodeFilter) Last() INode {
out := nf.execute(nf.Start)
if len(out) == 0 {
return nil
}
return out[len(out)-1]
}
// Get returns the Node at the given index in the NodeFilter; if index is invalid (<0 or >= len(nodes)), this function returns nil.
func (nf NodeFilter) Get(index int) INode {
out := nf.execute(nf.Start)
if index < 0 || index >= len(out) {
return nil
}
return out[index]
}
// ByFunc allows you to filter a given selection of nodes by the provided filter function (which takes a Node
// and returns a boolean, indicating whether or not to add that Node to the resulting NodeFilter).
// If no matching Nodes are found, an empty NodeFilter is returned.
func (nf NodeFilter) ByFunc(filterFunc func(node INode) bool) NodeFilter {
nf.Filters = append(nf.Filters, filterFunc)
return nf
}
// ByPropNameRegex allows you to filter a given selection of nodes by the provided set of property names,
// evaluated as regex strings.
// If the Nodes in the filter have any property that satisfy one of the supplied regex string, they pass
// the filter and are included.
// If no matching Nodes are found, an empty NodeFilter is returned.
func (nf NodeFilter) ByPropNameRegex(propNameRegexStrings ...string) NodeFilter {
nf.Filters = append(nf.Filters, func(node INode) bool {
for _, propName := range propNameRegexStrings {
for existingName := range node.Properties() {
match, _ := regexp.MatchString(propName, existingName)
if match {
return true
}
}
}
return false
})
return nf
}
// ByProp allows you to filter a given selection of nodes by a property value check - if the nodes filtered
// have a property with the given value, they are included.
// If no matching Nodes are found, an empty NodeFilter is returned.
func (nf NodeFilter) ByProp(propName string, propValue any) NodeFilter {
nf.Filters = append(nf.Filters, func(node INode) bool {
value := node.Properties().Get(propName)
return value != nil && value.Value == propValue
})
return nf
}
// ByParentProps allows you to filter a given selection of nodes if the node has a parent with the provided
// set of property names.
// If anyProp is true, then any matching Node will be added if it has any of the properties provided;
// otherwise, the Node would have to have all of the properties provided.
// If no matching Nodes are found, an empty NodeFilter is returned.
func (nf NodeFilter) ByParentProps(anyProp bool, propNames ...string) NodeFilter {
nf.Filters = append(nf.Filters, func(node INode) bool {
if anyProp {
if node.Parent() != nil {
parent := node.Parent()
for _, p := range propNames {
if parent.Properties().Has(p) {
return true
}
}
}
return false
}
return node.Parent() != nil && node.Parent().Properties().Has(propNames...)
})
return nf
}
// ByName allows you to filter a given selection of nodes if their names are wholly equal
// to the provided name string.
// If a Node's name doesn't match, it isn't added to the filter results.
func (nf NodeFilter) ByName(name string) NodeFilter {
nf.Filters = append(nf.Filters, func(node INode) bool { return node.Name() == name })
return nf
}
// ByRegex allows you to filter a given selection of nodes by their names using the given regex string.
// If you want to filter a selection of nodes in such a way that only nodes that have names that
// >contain< the given text (i.e. strings.Contains()) are selected for filtering, you can just pass
// that string into ByRegex directly.
// If the regexp string is invalid or no matching Nodes are found, the node isn't
// added to the filter results. See https://regexr.com/ for regex help / reference.
func (nf NodeFilter) ByRegex(regexString string) NodeFilter {
nf.Filters = append(nf.Filters, func(node INode) bool {
match, _ := regexp.MatchString(regexString, node.Name())
return match
})
return nf
}
// ByType allows you to filter a given selection of nodes by the provided NodeType.
// If no matching Nodes are found, an empty NodeFilter is returned.
func (nf NodeFilter) ByType(nodeType NodeType) NodeFilter {
nf.Filters = append(nf.Filters, func(node INode) bool {
return node.Type().Is(nodeType)
})
return nf
}
// Not allows you to filter OUT a given NodeFilter of nodes.
// If a node is in the provided slice of INodes, then it will not be added to the
// final NodeFilter.
func (nf NodeFilter) Not(others ...INode) NodeFilter {
nf.Filters = append(nf.Filters, func(node INode) bool {
for _, other := range others {
if node == other {
return false
}
}
return true
})
return nf
}
// StopOnFiltered allows you to specify that if a node doesn't pass a filter, none of its children will pass as well.
func (nf NodeFilter) StopOnFiltered() NodeFilter {
nf.stopOnFiltered = true
return nf
}
func (nf NodeFilter) bySectors() NodeFilter {
nf.Filters = append(nf.Filters, func(node INode) bool {
return node.Type() == NodeTypeModel && node.(*Model).sector != nil
})
return nf
}
// ForEach executes the provided function on each filtered Node on an internal slice; this is done to avoid allocating a slice for the
// filtered Nodes.
// The function must return a boolean indicating whether to continue running on each node in the tree that fulfills the
// filter set (true) or not (false).
// ForEach does not work with any sorting, and will log a warning if you use sorting and ForEach on the same filter.
func (nf NodeFilter) ForEach(callback func(node INode) bool) {
nf.executeFilters(nf.Start, callback, false)
}
// ForEachMultithreaded executes the provided function on each filtered Node on an internal slice; this is done to avoid allocating a
// slice for the filtered Nodes.
// ForEachMultithreaded will filter each node on the main thread, but will execute the callback function for each node
// on varying goroutines, syncing them all up once finished to resume execution on the main thread.
// You should be careful to avoid race conditions when using this function.
func (nf NodeFilter) ForEachMultithreaded(callback func(node INode)) {
nf.executeFilters(nf.Start, func(i INode) bool { callback(i); return true }, true)
}
// Count returns the number of Nodes that fit the filter set.
func (nf NodeFilter) Count() int {
count := 0
nf.executeFilters(nf.Start, func(i INode) bool {
count++
return true
}, false)
return count
}
// Contains returns if the provided Node is contained in the NodeFilter.
func (nf NodeFilter) Contains(node INode) bool {
return nf.Index(node) > 0
}
// Index returns the index of the given INode in the NodeFilter assuming the filter executes and results
// in an []INode. If it doesn't exist in the filter, then this function returns -1.
func (nf NodeFilter) Index(node INode) int {
out := nf.execute(nf.Start)
for index, child := range out {
if child == node {
return index
}
}
return -1
}
// IsZero returns true if the NodeFilter is uninitialized / empty.
func (nf NodeFilter) IsZero() bool {
return nf.Start == nil
}
// IsEmpty returns true if no Nodes pass the NodeFilter.
func (nf NodeFilter) IsEmpty() bool {
return len(nf.execute(nf.Start)) == 0
}
// INodes returns the NodeFilter's results as a slice of INodes.
func (nf NodeFilter) INodes() NodeCollection[INode] {
return nf.execute(nf.Start)
}
// IBoundingObjects returns a slice of the IBoundingObjects contained within the NodeFilter.
func (nf NodeFilter) IBoundingObjects() NodeCollection[IBoundingObject] {
out := nf.execute(nf.Start)
boundings := make([]IBoundingObject, 0, len(out))
for _, n := range out {
if b, ok := n.(IBoundingObject); ok {
boundings = append(boundings, b)
}
}
return boundings
}
// Models returns a slice of the Models contained within the NodeFilter.
func (nf NodeFilter) Models() NodeCollection[*Model] {
out := nf.execute(nf.Start)
models := make([]*Model, 0, len(out))
for _, n := range out {
if m, ok := n.(*Model); ok {
models = append(models, m)
}
}
return models
}
// ILights returns a slice of the ILights contained within the NodeFilter.
func (nf NodeFilter) ILights() NodeCollection[ILight] {
out := nf.execute(nf.Start)
lights := make([]ILight, 0, len(out))
for _, n := range out {
if m, ok := n.(ILight); ok {
lights = append(lights, m)
}
}
return lights
}
// Grids returns a slice of the Grid nodes contained within the NodeFilter.
func (nf NodeFilter) Grids() NodeCollection[*Grid] {
out := nf.execute(nf.Start)
grids := make([]*Grid, 0, len(out))
for _, n := range out {
if m, ok := n.(*Grid); ok {
grids = append(grids, m)
}
}
return grids
}
// SortByX applies an X-axis sort on the results of the NodeFilter.
// Sorts do not combine.
func (nf NodeFilter) SortByX() NodeFilter {
nf.sortMode = nfSortModeAxisX
return nf
}
// SortByY applies an Y-axis sort on the results of the NodeFilter.
// Sorts do not combine.
func (nf NodeFilter) SortByY() NodeFilter {
nf.sortMode = nfSortModeAxisY
return nf
}
// SortByZ applies an Z-axis sort on the results of the NodeFilter.
// Sorts do not combine.
func (nf NodeFilter) SortByZ() NodeFilter {
nf.sortMode = nfSortModeAxisZ
return nf
}
// SortByDistance sorts the results of the NodeFilter by their distances to the specified INode.
// Sorts do not combine.
func (nf NodeFilter) SortByDistance(to Vector3) NodeFilter {
nf.sortMode = nfSortModeDistance
nf.sortTo = to
return nf
}
// SortReverse reverses any sorting performed on the NodeFilter.
func (nf NodeFilter) SortReverse() NodeFilter {
nf.reverseSort = true
return nf
}
// SortReverse randomizes the elements returned by the NodeFilter finishing functions.
// Sorts do not combine.
func (nf NodeFilter) SortRandom() NodeFilter {
nf.sortMode = nfSortModeRandom
return nf
}
// SetDepth sets the maximum search depth of the NodeFilter to the value provided.
func (nf NodeFilter) SetMaxDepth(depth int) NodeFilter {
nf.MaxDepth = depth
return nf
}