< Summary

Class:KDTree[T]
Assembly:bamlab.micromissiles
File(s):/github/workspace/Assets/Scripts/Utils/KDTree.cs
Covered lines:43
Uncovered lines:0
Coverable lines:43
Total lines:87
Line coverage:100% (43 of 43)
Covered branches:0
Total branches:0
Covered methods:4
Total methods:4
Method coverage:100% (4 of 4)

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity NPath complexity Sequence coverage
KDTree(...)0%110100%
BuildTree(...)0%220100%
NearestNeighbor(...)0%220100%
NearestNeighbor(...)0%14140100%

File(s)

/github/workspace/Assets/Scripts/Utils/KDTree.cs

#LineLine coverage
 1using System;
 2using System.Collections.Generic;
 3using System.Linq;
 4using UnityEngine;
 5
 6// K-D tree node.
 7public class KDNode<T> {
 8  public T Data;
 9  public KDNode<T> Left = null;
 10  public KDNode<T> Right = null;
 11
 12  public KDNode(T data) {
 13    Data = data;
 14  }
 15}
 16
 17// K-D tree with 2D coordinates.
 18public class KDTree<T> {
 19  private KDNode<T> _root;
 20  private Func<T, Vector2> _getCoordinates;
 21
 3222  public KDTree(in IReadOnlyList<T> points, Func<T, Vector2> getCoordinates) {
 1623    _getCoordinates = getCoordinates;
 1624    _root = BuildTree(points, depth: 0);
 1625  }
 26
 98227  private KDNode<T> BuildTree(in IReadOnlyList<T> points, int depth) {
 98228    if (points.Count == 0)
 49929      return null;
 30
 48331    int k = 2;
 48332    int axis = depth % k;
 33
 34    // Sort the points by axis and find the median.
 48335    List<T> sortedPoints =
 279936        points.OrderBy(point => (axis == 0 ? _getCoordinates(point).x : _getCoordinates(point).y))
 37            .ToList();
 48338    int medianIndex = sortedPoints.Count / 2;
 48339    T medianPoint = sortedPoints[medianIndex];
 40
 48341    KDNode<T> node = new KDNode<T>(medianPoint);
 48342    List<T> leftPoints = sortedPoints.GetRange(0, medianIndex);
 48343    List<T> rightPoints =
 44        sortedPoints.GetRange(medianIndex + 1, sortedPoints.Count - medianIndex - 1);
 45
 48346    node.Left = BuildTree(leftPoints, depth + 1);
 48347    node.Right = BuildTree(rightPoints, depth + 1);
 48348    return node;
 98249  }
 50
 4551  public T NearestNeighbor(Vector2 target) {
 4552    KDNode<T> neighbor = NearestNeighbor(_root, target, depth: 0, best: null);
 4953    if (neighbor == null) {
 454      return default(T);
 55    }
 4156    return neighbor.Data;
 4557  }
 58
 58359  private KDNode<T> NearestNeighbor(KDNode<T> node, Vector2 target, int depth, KDNode<T> best) {
 58360    if (node == null)
 19361      return best;
 62
 39063    float currentDistance = Vector2.Distance(_getCoordinates(node.Data), target);
 39064    float bestDistance =
 65        best == null ? float.MaxValue : Vector2.Distance(_getCoordinates(best.Data), target);
 39066    if (currentDistance < bestDistance)
 10967      best = node;
 68
 39069    int axis = depth % 2;
 39070    KDNode<T> nextBranch = (axis == 0 ? target.x < _getCoordinates(node.Data).x
 71                                      : target.y < _getCoordinates(node.Data).y)
 72                               ? node.Left
 73                               : node.Right;
 39074    KDNode<T> otherBranch = (nextBranch == node.Left) ? node.Right : node.Left;
 75
 76    // Check the next branch.
 39077    best = NearestNeighbor(nextBranch, target, depth + 1, best);
 78
 79    // Check the other branch.
 39080    if ((axis == 0 && Mathf.Abs(target.x - _getCoordinates(node.Data).x) < bestDistance) ||
 14881        (axis == 1 && Mathf.Abs(target.y - _getCoordinates(node.Data).y) < bestDistance)) {
 14882      best = NearestNeighbor(otherBranch, target, depth + 1, best);
 14883    }
 84
 39085    return best;
 58386  }
 87}