コード例 #1
0
 private void PropagateNodeChange(AabbTreeNode <T> innernode, bool recalculateBounds)
 {
     while (innernode != null)
     {
         var parent = innernode.parent;
         if (innernode.children.Count <= 0)
         {
             ReplaceNode(innernode, null);
         }
         else if (parent != null && innernode.children.Count < minChildren)
         {
             ReplaceNode(innernode, null);
             foreach (var child in innernode.children)
             {
                 AddChildNode(parent, child);
             }
         }
         else
         {
             if (innernode.children.Count > maxChildren)
             {
                 innernode         = ExplodeNode(innernode);
                 recalculateBounds = true;
             }
             ShallowUpdateNode(innernode, recalculateBounds);
             BalanceNodeByRotations(innernode);
         }
         innernode = parent;
     }
 }
コード例 #2
0
        private void AddChildNode(AabbTreeNode <T> parent, AabbTreeNode <T> child)
        {
            if (parent?.children == null)
            {
                throw new InvalidOperationException("AddChild parent cannot be null");
            }
            if (child == null)
            {
                throw new InvalidOperationException("AddChild child cannot be null");
            }
            if (parent.leaf)
            {
                throw new InvalidOperationException("AddChild parent cannot be a leaf node");
            }
            var node = parent;

            while (node != null)
            {
                if (node == child)
                {
                    throw new InvalidOperationException("AddChild attempted to create circular graph");
                }
                node = node.parent;
            }
            parent.children.Add(child);
            child.parent = parent;
        }
コード例 #3
0
        private void SwapNodes(AabbTreeNode <T> a, AabbTreeNode <T> b)
        {
            AabbTreeNode <T> aparent = a.parent, bparent = b.parent;

            if (aparent == null || bparent == null || a == b)
            {
                return;
            }
            DetachNode(a);
            DetachNode(b);
            AddChildNode(bparent, a);
            AddChildNode(aparent, b);
        }
コード例 #4
0
 private void AddToNode(AabbTreeNode <T> inner, AabbTreeNode <T> leaf)
 {
     for (;;)
     {
         inner = FindChildNodeWithLeastOverlapInInner(inner, leaf.bounds);
         if (inner == null)
         {
             throw new InvalidOperationException();
         }
         if (inner.leaf)
         {
             AddChildNode(inner.parent, leaf);
             break;
         }
     }
     PropagateNodeChange(inner.parent, true);
 }
コード例 #5
0
        private int DebugCountLeaves(AabbTreeNode <T> node)
        {
            if (node == null)
            {
                return(0);
            }
            if (node.leaf)
            {
                return(1);
            }
            int result = 0;

            foreach (var child in node.children)
            {
                result += DebugCountLeaves(child);
            }
            return(result);
        }
コード例 #6
0
        private void ReplaceNode(AabbTreeNode <T> node, AabbTreeNode <T> replacement)
        {
            var parent = node?.parent;

            if (parent == null)
            {
                root = replacement;
                replacement.parent = null;
            }
            else
            {
                parent.children.Remove(node);
                if (replacement != null)
                {
                    AddChildNode(parent, replacement);
                }
            }
        }
コード例 #7
0
        private AabbTreeNode <T> RebuildNode(List <AabbTreeNode <T> > leaves, AabbTreeNode <T> parent)
        {
            int expectedChildren = maxChildren - 1;

            if (leaves.Count <= expectedChildren)
            {
                var n = new AabbTreeNode <T>
                {
                    children = leaves,
                    depth    = MaxDepth(leaves) + 1,
                    parent   = parent,
                };
                foreach (var leaf in leaves)
                {
                    leaf.parent = n;
                }
                return(n);
            }
            var unionrect = AaRects.MergeAll(leaves.Select(l => l.bounds));

            if (unionrect.GetSize().x > unionrect.GetSize().y)
            {
                leaves.Sort(OrderNodeByX);
            }
            else
            {
                leaves.Sort(OrderNodeByY);
            }
            var result = new AabbTreeNode <T>
            {
                children = new List <AabbTreeNode <T> >(),
                parent   = parent,
            };

            for (int i = 0; i < expectedChildren; i++)
            {
                var i0 = i * leaves.Count / expectedChildren;
                var i1 = (i + 1) * leaves.Count / expectedChildren;
                result.children.Add(RebuildNode(leaves.GetRange(i0, i1 - i0), result));
            }

            ShallowUpdateNode(result, true);
            return(result);
        }
コード例 #8
0
 private void AddToRoot(AabbTreeNode <T> leaf)
 {
     if (root == null)
     {
         root = new AabbTreeNode <T>
         {
             children = new List <AabbTreeNode <T> > {
                 leaf
             },
             depth  = leaf.depth + 1,
             bounds = leaf.bounds,
         };
         leaf.parent = root;
     }
     else
     {
         AddToNode(root, leaf);
     }
 }
コード例 #9
0
        public void Rebuild()
        {
            if (type != TreeType.RebuildTree)
            {
                return;
            }

            var list = new List <AabbTreeNode <T> >();

            foreach (var b in bodies)
            {
                var n = new AabbTreeNode <T>
                {
                    body   = b,
                    bounds = GetBodyFitRect(b),
                    leaf   = true,
                };
                bodyNodes[b] = n;
                list.Add(n);
            }
            root = RebuildNode(list, null);
        }
コード例 #10
0
        private AabbTreeNode <T> ExplodeNode(AabbTreeNode <T> node)
        {
            if (node.leaf)
            {
                return(node);
            }
            var size = node.bounds.GetSize();

            node.children.Sort(size.y > size.x ? OrderNodeByY : OrderNodeByX);
            var oldChildren = node.children;
            var childCount  = oldChildren.Count;

            node.children = new List <AabbTreeNode <T> >();
            var intermediatorCount = childCount / minChildren;

            for (var i = 0; i < intermediatorCount; i++)
            {
                AddChildNode(node, new AabbTreeNode <T>()
                {
                    children = new List <AabbTreeNode <T> >(),
                    parent   = node,
                    leaf     = false
                });
            }

            var childIndex = 0;

            foreach (var child in oldChildren)
            {
                var intermediate = node.children[childIndex * intermediatorCount / childCount];
                AddChildNode(intermediate, child);
                childIndex++;
            }
            foreach (var intermediator in node.children)
            {
                ShallowUpdateNode(intermediator, true);
            }
            return(node);
        }
コード例 #11
0
        private void PossibleCollisionsUsingTree(AaRect rect, AabbTreeNode <T> node, Action <T> action)
        {
            if (!AaRect.Overlaps(rect, node.bounds))
            {
                return;
            }

            marks?.Add(node);
            if (node.leaf)
            {
                if (AaRect.Overlaps(rect, GetBodyFitRect(node.body)))
                {
                    action?.Invoke(node.body);
                }
            }
            else
            {
                foreach (var child in node.children)
                {
                    PossibleCollisionsUsingTree(rect, child, action);
                }
            }
        }
コード例 #12
0
        private void ShallowUpdateNode(AabbTreeNode <T> node, bool recalculateBounds)
        {
            if (node.leaf)
            {
                return;
            }
            var maxnode = GetMaxDepthChild(node);

            if (node == null)
            {
                throw new InvalidOperationException();
            }

            if (maxnode == null)
            {
                throw new InvalidOperationException();
            }
            node.depth = maxnode.depth + 1;
            if (recalculateBounds)
            {
                node.bounds = AaRects.MergeAll(node.children.Select(c => c.bounds));
            }
        }
コード例 #13
0
        public override bool Add(T body)
        {
            if (type == TreeType.Bag)
            {
                return(bodies.Add(body));
            }

            AabbTreeNode <T> node;
            bool             result = bodyNodes.TryGetValue(body, out node);

            if (result)
            {
                if (node.bounds.Contains(GetBodyFitRect(body)))
                {
                    if (--node.fatsleep > 0)
                    {
                        return(true); // nothing to do, still inside fat bounds
                    }
                }
                RemoveNode(node);
            }
            else
            {
                node = new AabbTreeNode <T>
                {
                    leaf = true,
                    body = body,
                };
                bodyNodes[body] = node;
                bodies.Add(body);
            }
            node.bounds   = GetBodyFatRect(body);
            node.fatsleep = fatSleepFrames;
            AddToRoot(node);
            return(result);
        }
コード例 #14
0
 private void BalanceNodeByRotations(AabbTreeNode <T> node)
 {
     if (!balanceEnableRotations)
     {
         return;
     }
     if (node.leaf)
     {
         return;
     }
     for (var safetyi = 0; safetyi < 3; safetyi++)
     {
         var maxnode   = GetMaxDepthChild(node);
         var minnode   = GetMinDepthChild(node);
         var threshold = balanceDepthDifferenceThreshold + node.depth * balanceDepthDifferencePercent / 100;
         if (maxnode.depth <= minnode.depth + threshold)
         {
             break;
         }
         var maxnodechild = GetMaxDepthChild(maxnode);
         SwapNodes(maxnodechild, minnode);
         ShallowUpdateNode(maxnode, true);
     }
 }
コード例 #15
0
 private void DetachNode(AabbTreeNode <T> node)
 {
     node.parent.children.Remove(node);
     node.parent = null;
 }
コード例 #16
0
 private AabbTreeNode <T> FindChildNodeWithLeastOverlapInInner(AabbTreeNode <T> node, AaRect fat)
 {
     return(node.children.MinBy <AabbTreeNode <T>, List <AabbTreeNode <T> >, float>(
                n => AaRect.Merge(n.bounds, fat).GetArea() - n.bounds.GetArea()
                ));
 }
コード例 #17
0
 private AabbTreeNode <T> GetMaxDepthChild(AabbTreeNode <T> node)
 {
     return(node.children.MaxBy(DepthOf));
 }
コード例 #18
0
 private void RemoveNode(AabbTreeNode <T> leaf)
 {
     leaf.parent.children.Remove(leaf);
     PropagateNodeChange(leaf.parent, true);
 }