Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(105)

Unified Diff: trunk/src/ui/gfx/geometry/r_tree_unittest.cc

Issue 338833004: Revert 276827 "readability review for luken" (Closed) Base URL: svn://svn.chromium.org/chrome/
Patch Set: Created 6 years, 6 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « trunk/src/ui/gfx/geometry/r_tree_base.cc ('k') | trunk/src/ui/gfx/gfx.gyp » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: trunk/src/ui/gfx/geometry/r_tree_unittest.cc
===================================================================
--- trunk/src/ui/gfx/geometry/r_tree_unittest.cc (revision 277531)
+++ trunk/src/ui/gfx/geometry/r_tree_unittest.cc (working copy)
@@ -4,344 +4,459 @@
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/gfx/geometry/r_tree.h"
-#include "ui/gfx/geometry/r_tree_base.h"
#include "ui/gfx/geometry/rect.h"
namespace gfx {
class RTreeTest : public ::testing::Test {
protected:
- typedef RTree<int> RT;
-
- // Given a pointer to an RTree, traverse it and verify that its internal
- // structure is consistent with RTree semantics.
- void ValidateRTree(RTreeBase* rt) {
+ // Given a pointer to an RTree, traverse it and verify its internal structure
+ // is consistent with the RTree semantics.
+ void ValidateRTree(RTree* rt) {
// If RTree is empty it should have an empty rectangle.
- if (!rt->root()->count()) {
- EXPECT_TRUE(rt->root()->rect().IsEmpty());
- EXPECT_EQ(0, rt->root()->Level());
+ if (!rt->root_->count()) {
+ EXPECT_TRUE(rt->root_->rect().IsEmpty());
+ EXPECT_EQ(rt->root_->level(), 0);
return;
}
// Root is allowed to have fewer than min_children_ but never more than
// max_children_.
- EXPECT_LE(rt->root()->count(), rt->max_children_);
+ EXPECT_LE(rt->root_->count(), rt->max_children_);
// The root should never be a record node.
- EXPECT_GT(rt->root()->Level(), -1);
+ EXPECT_GT(rt->root_->level(), -1);
+ EXPECT_FALSE(rt->root_->key());
// The root should never have a parent pointer.
- EXPECT_TRUE(rt->root()->parent() == NULL);
+ EXPECT_FALSE(rt->root_->parent());
// Bounds must be consistent on the root.
- CheckBoundsConsistent(rt->root());
- for (size_t i = 0; i < rt->root()->count(); ++i) {
+ CheckBoundsConsistent(rt->root_.get());
+ // We traverse root's children ourselves, so we can avoid asserts about
+ // root's potential inconsistencies.
+ for (size_t i = 0; i < rt->root_->children_.size(); ++i) {
ValidateNode(
- rt->root()->child(i), rt->min_children_, rt->max_children_);
+ rt->root_->children_[i], rt->min_children_, rt->max_children_);
}
}
// Recursive descent method used by ValidateRTree to check each node within
// the RTree for consistency with RTree semantics.
- void ValidateNode(const RTreeBase::NodeBase* node_base,
+ void ValidateNode(RTree::Node* node,
size_t min_children,
size_t max_children) {
- if (node_base->Level() >= 0) {
- const RTreeBase::Node* node =
- static_cast<const RTreeBase::Node*>(node_base);
- EXPECT_GE(node->count(), min_children);
- EXPECT_LE(node->count(), max_children);
- CheckBoundsConsistent(node);
- for (size_t i = 0; i < node->count(); ++i)
- ValidateNode(node->child(i), min_children, max_children);
+ // Record nodes have different requirements, handle up front.
+ if (node->level() == -1) {
+ // Record nodes may have no children.
+ EXPECT_EQ(node->count(), 0U);
+ // They must have an associated non-NULL key.
+ EXPECT_TRUE(node->key());
+ // They must always have a parent.
+ EXPECT_TRUE(node->parent());
+ return;
}
+ // Non-record node, normal expectations apply.
+ EXPECT_GE(node->count(), min_children);
+ EXPECT_LE(node->count(), max_children);
+ EXPECT_EQ(node->key(), 0);
+ CheckBoundsConsistent(node);
+ for (size_t i = 0; i < node->children_.size(); ++i) {
+ ValidateNode(node->children_[i], min_children, max_children);
+ }
}
// Check bounds are consistent with children bounds, and other checks
// convenient to do while enumerating the children of node.
- void CheckBoundsConsistent(const RTreeBase::Node* node) {
- EXPECT_FALSE(node->rect().IsEmpty());
+ void CheckBoundsConsistent(RTree::Node* node) {
+ EXPECT_FALSE(node->rect_.IsEmpty());
Rect check_bounds;
- for (size_t i = 0; i < node->count(); ++i) {
- const RTreeBase::NodeBase* child_node = node->child(i);
+ for (size_t i = 0; i < node->children_.size(); ++i) {
+ RTree::Node* child_node = node->children_[i];
check_bounds.Union(child_node->rect());
- EXPECT_EQ(node->Level() - 1, child_node->Level());
+ EXPECT_EQ(node->level() - 1, child_node->level());
EXPECT_EQ(node, child_node->parent());
}
- EXPECT_EQ(check_bounds, node->rect());
+ EXPECT_EQ(node->rect_, check_bounds);
}
// Adds count squares stacked around the point (0,0) with key equal to width.
- void AddStackedSquares(RT* rt, int count) {
+ void AddStackedSquares(RTree* rt, int count) {
for (int i = 1; i <= count; ++i) {
rt->Insert(Rect(0, 0, i, i), i);
- ValidateRTree(static_cast<RTreeBase*>(rt));
+ ValidateRTree(rt);
}
}
- // Given an unordered list of matching keys, verifies that it contains all
+ // Given an unordered list of matching keys, verify that it contains all
// values [1..length] for the length of that list.
- void VerifyAllKeys(const RT::Matches& keys) {
- for (size_t i = 1; i <= keys.size(); ++i)
- EXPECT_EQ(1U, keys.count(i));
+ void VerifyAllKeys(const base::hash_set<intptr_t>& keys) {
+ // Verify that the keys are in values [1,count].
+ for (size_t i = 1; i <= keys.size(); ++i) {
+ EXPECT_EQ(keys.count(i), 1U);
+ }
}
// Given a node and a rectangle, builds an expanded rectangle list where the
- // ith element of the vector is the union of the rectangle of the ith child of
+ // ith element of the rectangle is union of the recangle of the ith child of
// the node and the argument rectangle.
- void BuildExpandedRects(RTreeBase::Node* node,
+ void BuildExpandedRects(RTree::Node* node,
const Rect& rect,
std::vector<Rect>* expanded_rects) {
expanded_rects->clear();
- expanded_rects->reserve(node->count());
- for (size_t i = 0; i < node->count(); ++i) {
+ expanded_rects->reserve(node->children_.size());
+ for (size_t i = 0; i < node->children_.size(); ++i) {
Rect expanded_rect(rect);
- expanded_rect.Union(node->child(i)->rect());
+ expanded_rect.Union(node->children_[i]->rect_);
expanded_rects->push_back(expanded_rect);
}
}
};
-class RTreeNodeTest : public RTreeTest {
- protected:
- typedef RTreeBase::NodeBase RTreeNodeBase;
- typedef RT::Record RTreeRecord;
- typedef RTreeBase::Node RTreeNode;
- typedef RTreeBase::Node::Rects RTreeRects;
- typedef RTreeBase::Nodes RTreeNodes;
+// An empty RTree should never return Query results, and RTrees should be empty
+// upon construction.
+TEST_F(RTreeTest, QueryEmptyTree) {
+ RTree rt(2, 10);
+ ValidateRTree(&rt);
+ base::hash_set<intptr_t> results;
+ Rect test_rect(25, 25);
+ rt.Query(test_rect, &results);
+ EXPECT_EQ(results.size(), 0U);
+ ValidateRTree(&rt);
+}
- // Accessors to private members of RTree::Node.
- const RTreeRecord* record(RTreeNode* node, size_t i) {
- return static_cast<const RTreeRecord*>(node->child(i));
- }
+// Clear should empty the tree, meaning that all queries should not return
+// results after.
+TEST_F(RTreeTest, ClearEmptiesTreeOfSingleNode) {
+ RTree rt(2, 5);
+ rt.Insert(Rect(0, 0, 100, 100), 1);
+ rt.Clear();
+ base::hash_set<intptr_t> results;
+ Rect test_rect(1, 1);
+ rt.Query(test_rect, &results);
+ EXPECT_EQ(results.size(), 0U);
+ ValidateRTree(&rt);
+}
- // Provides access for tests to private methods of RTree::Node.
- scoped_ptr<RTreeNode> NewNodeAtLevel(size_t level) {
- return make_scoped_ptr(new RTreeBase::Node(level));
- }
+// Even with a complex internal structure, clear should empty the tree, meaning
+// that all queries should not return results after.
+TEST_F(RTreeTest, ClearEmptiesTreeOfManyNodes) {
+ RTree rt(2, 5);
+ AddStackedSquares(&rt, 100);
+ rt.Clear();
+ base::hash_set<intptr_t> results;
+ Rect test_rect(1, 1);
+ rt.Query(test_rect, &results);
+ EXPECT_EQ(results.size(), 0U);
+ ValidateRTree(&rt);
+}
- void NodeRecomputeLocalBounds(RTreeNodeBase* node) {
- node->RecomputeLocalBounds();
+// Duplicate inserts should overwrite previous inserts.
+TEST_F(RTreeTest, DuplicateInsertsOverwrite) {
+ RTree rt(2, 5);
+ // Add 100 stacked squares, but always with duplicate key of 1.
+ for (int i = 1; i <= 100; ++i) {
+ rt.Insert(Rect(0, 0, i, i), 1);
+ ValidateRTree(&rt);
}
+ base::hash_set<intptr_t> results;
+ Rect test_rect(1, 1);
+ rt.Query(test_rect, &results);
+ EXPECT_EQ(results.size(), 1U);
+ EXPECT_EQ(results.count(1), 1U);
+}
- bool NodeCompareVertical(RTreeNodeBase* a, RTreeNodeBase* b) {
- return RTreeBase::Node::CompareVertical(a, b);
+// Call Remove() once on something that's been inserted repeatedly.
+TEST_F(RTreeTest, DuplicateInsertRemove) {
+ RTree rt(3, 9);
+ AddStackedSquares(&rt, 25);
+ for (int i = 1; i <= 100; ++i) {
+ rt.Insert(Rect(0, 0, i, i), 26);
+ ValidateRTree(&rt);
}
+ rt.Remove(26);
+ base::hash_set<intptr_t> results;
+ Rect test_rect(1, 1);
+ rt.Query(test_rect, &results);
+ EXPECT_EQ(results.size(), 25U);
+ VerifyAllKeys(results);
+}
- bool NodeCompareHorizontal(RTreeNodeBase* a, RTreeNodeBase* b) {
- return RTreeBase::Node::CompareHorizontal(a, b);
+// Call Remove() repeatedly on something that's been inserted once.
+TEST_F(RTreeTest, InsertDuplicateRemove) {
+ RTree rt(7, 15);
+ AddStackedSquares(&rt, 101);
+ for (int i = 0; i < 100; ++i) {
+ rt.Remove(101);
+ ValidateRTree(&rt);
}
+ base::hash_set<intptr_t> results;
+ Rect test_rect(1, 1);
+ rt.Query(test_rect, &results);
+ EXPECT_EQ(results.size(), 100U);
+ VerifyAllKeys(results);
+}
- bool NodeCompareCenterDistanceFromParent(
- const RTreeNodeBase* a, const RTreeNodeBase* b) {
- return RTreeBase::Node::CompareCenterDistanceFromParent(a, b);
- }
+// Stacked rects should meet all matching queries regardless of nesting.
+TEST_F(RTreeTest, QueryStackedSquaresNestedHit) {
+ RTree rt(2, 5);
+ AddStackedSquares(&rt, 100);
+ base::hash_set<intptr_t> results;
+ Rect test_rect(1, 1);
+ rt.Query(test_rect, &results);
+ EXPECT_EQ(results.size(), 100U);
+ VerifyAllKeys(results);
+}
- int NodeOverlapIncreaseToAdd(RTreeNode* node,
- const Rect& rect,
- const RTreeNodeBase* candidate_node,
- const Rect& expanded_rect) {
- return node->OverlapIncreaseToAdd(rect, candidate_node, expanded_rect);
- }
+// Stacked rects should meet all matching queries when contained completely by
+// the query rectangle.
+TEST_F(RTreeTest, QueryStackedSquaresContainedHit) {
+ RTree rt(2, 10);
+ AddStackedSquares(&rt, 100);
+ base::hash_set<intptr_t> results;
+ Rect test_rect(0, 0, 100, 100);
+ rt.Query(test_rect, &results);
+ EXPECT_EQ(results.size(), 100U);
+ VerifyAllKeys(results);
+}
- void NodeBuildLowBounds(const std::vector<RTreeNodeBase*>& vertical_sort,
- const std::vector<RTreeNodeBase*>& horizontal_sort,
- RTreeRects* vertical_bounds,
- RTreeRects* horizontal_bounds) {
- RTreeBase::Node::BuildLowBounds(
- vertical_sort, horizontal_sort, vertical_bounds, horizontal_bounds);
- }
+// Stacked rects should miss a missing query when the query has no intersection
+// with the rects.
+TEST_F(RTreeTest, QueryStackedSquaresCompleteMiss) {
+ RTree rt(2, 7);
+ AddStackedSquares(&rt, 100);
+ base::hash_set<intptr_t> results;
+ Rect test_rect(150, 150, 100, 100);
+ rt.Query(test_rect, &results);
+ EXPECT_EQ(results.size(), 0U);
+}
- void NodeBuildHighBounds(const std::vector<RTreeNodeBase*>& vertical_sort,
- const std::vector<RTreeNodeBase*>& horizontal_sort,
- RTreeRects* vertical_bounds,
- RTreeRects* horizontal_bounds) {
- RTreeBase::Node::BuildHighBounds(
- vertical_sort, horizontal_sort, vertical_bounds, horizontal_bounds);
+// Removing half the nodes after insertion should still result in a valid tree.
+TEST_F(RTreeTest, RemoveHalfStackedRects) {
+ RTree rt(2, 11);
+ AddStackedSquares(&rt, 200);
+ for (int i = 101; i <= 200; ++i) {
+ rt.Remove(i);
+ ValidateRTree(&rt);
}
-
- int NodeSmallestMarginSum(size_t start_index,
- size_t end_index,
- const RTreeRects& low_bounds,
- const RTreeRects& high_bounds) {
- return RTreeBase::Node::SmallestMarginSum(
- start_index, end_index, low_bounds, high_bounds);
+ base::hash_set<intptr_t> results;
+ Rect test_rect(1, 1);
+ rt.Query(test_rect, &results);
+ EXPECT_EQ(results.size(), 100U);
+ VerifyAllKeys(results);
+ // Add the nodes back in.
+ for (int i = 101; i <= 200; ++i) {
+ rt.Insert(Rect(0, 0, i, i), i);
+ ValidateRTree(&rt);
}
+ results.clear();
+ rt.Query(test_rect, &results);
+ EXPECT_EQ(results.size(), 200U);
+ VerifyAllKeys(results);
+}
- size_t NodeChooseSplitIndex(size_t min_children,
- size_t max_children,
- const RTreeRects& low_bounds,
- const RTreeRects& high_bounds) {
- return RTreeBase::Node::ChooseSplitIndex(
- min_children, max_children, low_bounds, high_bounds);
- }
+TEST_F(RTreeTest, InsertDupToRoot) {
+ RTree rt(2, 5);
+ rt.Insert(Rect(0, 0, 1, 2), 1);
+ ValidateRTree(&rt);
+ rt.Insert(Rect(0, 0, 2, 1), 1);
+ ValidateRTree(&rt);
+}
- scoped_ptr<RTreeNodeBase> NodeDivideChildren(
- RTreeNode* node,
- const RTreeRects& low_bounds,
- const RTreeRects& high_bounds,
- const std::vector<RTreeNodeBase*>& sorted_children,
- size_t split_index) {
- return node->DivideChildren(
- low_bounds, high_bounds, sorted_children, split_index);
+TEST_F(RTreeTest, InsertNegativeCoordsRect) {
+ RTree rt(5, 11);
+ for (int i = 1; i <= 100; ++i) {
+ rt.Insert(Rect(-i, -i, i, i), (i * 2) - 1);
+ ValidateRTree(&rt);
+ rt.Insert(Rect(0, 0, i, i), i * 2);
+ ValidateRTree(&rt);
}
+ base::hash_set<intptr_t> results;
+ Rect test_rect(-1, -1, 2, 2);
+ rt.Query(test_rect, &results);
+ EXPECT_EQ(results.size(), 200U);
+ VerifyAllKeys(results);
+}
- RTreeNode* NodeLeastOverlapIncrease(RTreeNode* node,
- const Rect& node_rect,
- const RTreeRects& expanded_rects) {
- return node->LeastOverlapIncrease(node_rect, expanded_rects);
+TEST_F(RTreeTest, RemoveNegativeCoordsRect) {
+ RTree rt(7, 21);
+ // Add 100 positive stacked squares.
+ AddStackedSquares(&rt, 100);
+ // Now add 100 negative stacked squares.
+ for (int i = 101; i <= 200; ++i) {
+ rt.Insert(Rect(100 - i, 100 - i, i - 100, i - 100), 301 - i);
+ ValidateRTree(&rt);
}
-
- RTreeNode* NodeLeastAreaEnlargement(RTreeNode* node,
- const Rect& node_rect,
- const RTreeRects& expanded_rects) {
- return node->LeastAreaEnlargement(node_rect, expanded_rects);
+ // Now remove half of the negative squares.
+ for (int i = 101; i <= 150; ++i) {
+ rt.Remove(301 - i);
+ ValidateRTree(&rt);
}
-};
+ // Queries should return 100 positive and 50 negative stacked squares.
+ base::hash_set<intptr_t> results;
+ Rect test_rect(-1, -1, 2, 2);
+ rt.Query(test_rect, &results);
+ EXPECT_EQ(results.size(), 150U);
+ VerifyAllKeys(results);
+}
-// RTreeNodeTest --------------------------------------------------------------
+TEST_F(RTreeTest, InsertEmptyRectReplacementRemovesKey) {
+ RTree rt(10, 31);
+ AddStackedSquares(&rt, 50);
+ ValidateRTree(&rt);
-TEST_F(RTreeNodeTest, RemoveNodesForReinsert) {
+ // Replace last square with empty rect.
+ rt.Insert(Rect(), 50);
+ ValidateRTree(&rt);
+
+ // Now query large area to get all rects in tree.
+ base::hash_set<intptr_t> results;
+ Rect test_rect(0, 0, 100, 100);
+ rt.Query(test_rect, &results);
+
+ // Should only be 49 rects in tree.
+ EXPECT_EQ(results.size(), 49U);
+ VerifyAllKeys(results);
+}
+
+TEST_F(RTreeTest, NodeRemoveNodesForReinsert) {
// Make a leaf node for testing removal from.
- scoped_ptr<RTreeNode> test_node(new RTreeNode);
+ scoped_ptr<RTree::Node> test_node(new RTree::Node(0));
// Build 20 record nodes with rectangle centers going from (1,1) to (20,20)
- for (int i = 1; i <= 20; ++i)
- test_node->AddChild(scoped_ptr<RTreeNodeBase>(
- new RTreeRecord(Rect(i - 1, i - 1, 2, 2), i)));
-
+ for (int i = 1; i <= 20; ++i) {
+ test_node->AddChild(new RTree::Node(Rect(i - 1, i - 1, 2, 2), i));
+ }
// Quick verification of the node before removing children.
ValidateNode(test_node.get(), 1U, 20U);
// Use a scoped vector to delete all children that get removed from the Node.
- RTreeNodes removals;
+ ScopedVector<RTree::Node> removals;
test_node->RemoveNodesForReinsert(1, &removals);
- // Should have gotten back 1 node pointer.
- EXPECT_EQ(1U, removals.size());
+ // Should have gotten back 1 node pointers.
+ EXPECT_EQ(removals.size(), 1U);
// There should be 19 left in the test_node.
- EXPECT_EQ(19U, test_node->count());
+ EXPECT_EQ(test_node->count(), 19U);
// If we fix up the bounds on the test_node, it should verify.
- NodeRecomputeLocalBounds(test_node.get());
+ test_node->RecomputeBoundsNoParents();
ValidateNode(test_node.get(), 2U, 20U);
// The node we removed should be node 10, as it was exactly in the center.
- EXPECT_EQ(10, static_cast<RTreeRecord*>(removals[0])->key());
+ EXPECT_EQ(removals[0]->key(), 10);
// Now remove the next 2.
removals.clear();
test_node->RemoveNodesForReinsert(2, &removals);
- EXPECT_EQ(2U, removals.size());
- EXPECT_EQ(17U, test_node->count());
- NodeRecomputeLocalBounds(test_node.get());
+ EXPECT_EQ(removals.size(), 2U);
+ EXPECT_EQ(test_node->count(), 17U);
+ test_node->RecomputeBoundsNoParents();
ValidateNode(test_node.get(), 2U, 20U);
// Lastly the 2 nodes we should have gotten back are keys 9 and 11, as their
// centers were the closest to the center of the node bounding box.
base::hash_set<intptr_t> results_hash;
- results_hash.insert(static_cast<RTreeRecord*>(removals[0])->key());
- results_hash.insert(static_cast<RTreeRecord*>(removals[1])->key());
- EXPECT_EQ(1U, results_hash.count(9));
- EXPECT_EQ(1U, results_hash.count(11));
+ results_hash.insert(removals[0]->key());
+ results_hash.insert(removals[1]->key());
+ EXPECT_EQ(results_hash.count(9), 1U);
+ EXPECT_EQ(results_hash.count(11), 1U);
}
-TEST_F(RTreeNodeTest, CompareVertical) {
- // One rect with lower y than another should always sort lower.
- RTreeRecord low(Rect(0, 1, 10, 10), 1);
- RTreeRecord middle(Rect(0, 5, 10, 10), 5);
- EXPECT_TRUE(NodeCompareVertical(&low, &middle));
- EXPECT_FALSE(NodeCompareVertical(&middle, &low));
+TEST_F(RTreeTest, NodeCompareVertical) {
+ // One rect with lower y than another should always sort lower than higher y.
+ RTree::Node low(Rect(0, 1, 10, 10), 1);
+ RTree::Node middle(Rect(0, 5, 10, 10), 5);
+ EXPECT_TRUE(RTree::Node::CompareVertical(&low, &middle));
+ EXPECT_FALSE(RTree::Node::CompareVertical(&middle, &low));
// Try a non-overlapping higher-y rectangle.
- RTreeRecord high(Rect(-10, 20, 10, 1), 10);
- EXPECT_TRUE(NodeCompareVertical(&low, &high));
- EXPECT_FALSE(NodeCompareVertical(&high, &low));
+ RTree::Node high(Rect(-10, 20, 10, 1), 10);
+ EXPECT_TRUE(RTree::Node::CompareVertical(&low, &high));
+ EXPECT_FALSE(RTree::Node::CompareVertical(&high, &low));
// Ties are broken by lowest bottom y value.
- RTreeRecord shorter_tie(Rect(10, 1, 100, 2), 2);
- EXPECT_TRUE(NodeCompareVertical(&shorter_tie, &low));
- EXPECT_FALSE(NodeCompareVertical(&low, &shorter_tie));
+ RTree::Node shorter_tie(Rect(10, 1, 100, 2), 2);
+ EXPECT_TRUE(RTree::Node::CompareVertical(&shorter_tie, &low));
+ EXPECT_FALSE(RTree::Node::CompareVertical(&low, &shorter_tie));
}
-TEST_F(RTreeNodeTest, CompareHorizontal) {
+TEST_F(RTreeTest, NodeCompareHorizontal) {
// One rect with lower x than another should always sort lower than higher x.
- RTreeRecord low(Rect(1, 0, 10, 10), 1);
- RTreeRecord middle(Rect(5, 0, 10, 10), 5);
- EXPECT_TRUE(NodeCompareHorizontal(&low, &middle));
- EXPECT_FALSE(NodeCompareHorizontal(&middle, &low));
+ RTree::Node low(Rect(1, 0, 10, 10), 1);
+ RTree::Node middle(Rect(5, 0, 10, 10), 5);
+ EXPECT_TRUE(RTree::Node::CompareHorizontal(&low, &middle));
+ EXPECT_FALSE(RTree::Node::CompareHorizontal(&middle, &low));
// Try a non-overlapping higher-x rectangle.
- RTreeRecord high(Rect(20, -10, 1, 10), 10);
- EXPECT_TRUE(NodeCompareHorizontal(&low, &high));
- EXPECT_FALSE(NodeCompareHorizontal(&high, &low));
+ RTree::Node high(Rect(20, -10, 1, 10), 10);
+ EXPECT_TRUE(RTree::Node::CompareHorizontal(&low, &high));
+ EXPECT_FALSE(RTree::Node::CompareHorizontal(&high, &low));
// Ties are broken by lowest bottom x value.
- RTreeRecord shorter_tie(Rect(1, 10, 2, 100), 2);
- EXPECT_TRUE(NodeCompareHorizontal(&shorter_tie, &low));
- EXPECT_FALSE(NodeCompareHorizontal(&low, &shorter_tie));
+ RTree::Node shorter_tie(Rect(1, 10, 2, 100), 2);
+ EXPECT_TRUE(RTree::Node::CompareHorizontal(&shorter_tie, &low));
+ EXPECT_FALSE(RTree::Node::CompareHorizontal(&low, &shorter_tie));
}
-TEST_F(RTreeNodeTest, CompareCenterDistanceFromParent) {
+TEST_F(RTreeTest, NodeCompareCenterDistanceFromParent) {
// Create a test node we can add children to, for distance comparisons.
- scoped_ptr<RTreeNode> parent(new RTreeNode);
+ scoped_ptr<RTree::Node> parent(new RTree::Node(0));
// Add three children, one each with centers at (0, 0), (10, 10), (-9, -9),
// around which a bounding box will be centered at (0, 0)
- scoped_ptr<RTreeRecord> center_zero(new RTreeRecord(Rect(-1, -1, 2, 2), 1));
- parent->AddChild(center_zero.PassAs<RTreeNodeBase>());
+ RTree::Node* center_zero = new RTree::Node(Rect(-1, -1, 2, 2), 1);
+ parent->AddChild(center_zero);
- scoped_ptr<RTreeRecord> center_positive(new RTreeRecord(Rect(9, 9, 2, 2), 2));
- parent->AddChild(center_positive.PassAs<RTreeNodeBase>());
+ RTree::Node* center_positive = new RTree::Node(Rect(9, 9, 2, 2), 2);
+ parent->AddChild(center_positive);
- scoped_ptr<RTreeRecord> center_negative(
- new RTreeRecord(Rect(-10, -10, 2, 2), 3));
- parent->AddChild(center_negative.PassAs<RTreeNodeBase>());
+ RTree::Node* center_negative = new RTree::Node(Rect(-10, -10, 2, 2), 3);
+ parent->AddChild(center_negative);
ValidateNode(parent.get(), 1U, 5U);
- EXPECT_EQ(Rect(-10, -10, 21, 21), parent->rect());
+ EXPECT_EQ(parent->rect_, Rect(-10, -10, 21, 21));
- EXPECT_TRUE(
- NodeCompareCenterDistanceFromParent(parent->child(0), parent->child(1)));
- EXPECT_FALSE(
- NodeCompareCenterDistanceFromParent(parent->child(1), parent->child(0)));
- EXPECT_TRUE(
- NodeCompareCenterDistanceFromParent(parent->child(0), parent->child(2)));
- EXPECT_FALSE(
- NodeCompareCenterDistanceFromParent(parent->child(2), parent->child(0)));
- EXPECT_TRUE(
- NodeCompareCenterDistanceFromParent(parent->child(2), parent->child(1)));
- EXPECT_FALSE(
- NodeCompareCenterDistanceFromParent(parent->child(1), parent->child(2)));
+ EXPECT_TRUE(RTree::Node::CompareCenterDistanceFromParent(center_zero,
+ center_positive));
+ EXPECT_FALSE(RTree::Node::CompareCenterDistanceFromParent(center_positive,
+ center_zero));
+
+ EXPECT_TRUE(RTree::Node::CompareCenterDistanceFromParent(center_zero,
+ center_negative));
+ EXPECT_FALSE(RTree::Node::CompareCenterDistanceFromParent(center_negative,
+ center_zero));
+
+ EXPECT_TRUE(RTree::Node::CompareCenterDistanceFromParent(center_negative,
+ center_positive));
+ EXPECT_FALSE(RTree::Node::CompareCenterDistanceFromParent(center_positive,
+ center_negative));
}
-TEST_F(RTreeNodeTest, OverlapIncreaseToAdd) {
+TEST_F(RTreeTest, NodeOverlapIncreaseToAdd) {
// Create a test node with three children, for overlap comparisons.
- scoped_ptr<RTreeNode> parent(new RTreeNode);
+ scoped_ptr<RTree::Node> parent(new RTree::Node(0));
// Add three children, each 4 wide and tall, at (0, 0), (3, 3), (6, 6) with
// overlapping corners.
Rect top(0, 0, 4, 4);
- parent->AddChild(scoped_ptr<RTreeNodeBase>(new RTreeRecord(top, 1)));
+ parent->AddChild(new RTree::Node(top, 1));
Rect middle(3, 3, 4, 4);
- parent->AddChild(scoped_ptr<RTreeNodeBase>(new RTreeRecord(middle, 2)));
+ parent->AddChild(new RTree::Node(middle, 2));
Rect bottom(6, 6, 4, 4);
- parent->AddChild(scoped_ptr<RTreeNodeBase>(new RTreeRecord(bottom, 3)));
+ parent->AddChild(new RTree::Node(bottom, 3));
ValidateNode(parent.get(), 1U, 5U);
// Test a rect in corner.
Rect corner(0, 0, 1, 1);
Rect expanded = top;
expanded.Union(corner);
- // It should not add any overlap to add this to the first child at (0, 0).
- EXPECT_EQ(0, NodeOverlapIncreaseToAdd(
- parent.get(), corner, parent->child(0), expanded));
+ // It should not add any overlap to add this to the first child at (0, 0);
+ EXPECT_EQ(parent->OverlapIncreaseToAdd(corner, 0, expanded), 0);
expanded = middle;
expanded.Union(corner);
// Overlap for middle rectangle should increase from 2 pixels at (3, 3) and
// (6, 6) to 17 pixels, as it will now cover 4x4 rectangle top,
- // so a change of +15.
- EXPECT_EQ(15, NodeOverlapIncreaseToAdd(
- parent.get(), corner, parent->child(1), expanded));
+ // so a change of +15
+ EXPECT_EQ(parent->OverlapIncreaseToAdd(corner, 1, expanded), 15);
expanded = bottom;
expanded.Union(corner);
// Overlap for bottom rectangle should increase from 1 pixel at (6, 6) to
// 32 pixels, as it will now cover both 4x4 rectangles top and middle,
- // so a change of 31.
- EXPECT_EQ(31, NodeOverlapIncreaseToAdd(
- parent.get(), corner, parent->child(2), expanded));
+ // so a change of 31
+ EXPECT_EQ(parent->OverlapIncreaseToAdd(corner, 2, expanded), 31);
// Test a rect that doesn't overlap with anything, in the far right corner.
Rect far_corner(9, 0, 1, 1);
@@ -349,61 +464,58 @@
expanded.Union(far_corner);
// Overlap of top should go from 1 to 4, as it will now cover the entire first
// row of pixels in middle.
- EXPECT_EQ(3, NodeOverlapIncreaseToAdd(
- parent.get(), far_corner, parent->child(0), expanded));
+ EXPECT_EQ(parent->OverlapIncreaseToAdd(far_corner, 0, expanded), 3);
expanded = middle;
expanded.Union(far_corner);
// Overlap of middle should go from 2 to 8, as it will cover the rightmost 4
- // pixels of top and the top 4 pixels of bottom as it expands.
- EXPECT_EQ(6, NodeOverlapIncreaseToAdd(
- parent.get(), far_corner, parent->child(1), expanded));
+ // pixels of top and the top 4 pixles of bottom as it expands.
+ EXPECT_EQ(parent->OverlapIncreaseToAdd(far_corner, 1, expanded), 6);
expanded = bottom;
expanded.Union(far_corner);
// Overlap of bottom should go from 1 to 4, as it will now cover the rightmost
// 4 pixels of middle.
- EXPECT_EQ(3, NodeOverlapIncreaseToAdd(
- parent.get(), far_corner, parent->child(2), expanded));
+ EXPECT_EQ(parent->OverlapIncreaseToAdd(far_corner, 2, expanded), 3);
}
-TEST_F(RTreeNodeTest, BuildLowBounds) {
- RTreeNodes records;
- records.reserve(10);
- for (int i = 1; i <= 10; ++i)
- records.push_back(new RTreeRecord(Rect(0, 0, i, i), i));
-
- RTreeRects vertical_bounds;
- RTreeRects horizontal_bounds;
- NodeBuildLowBounds(
- records.get(), records.get(), &vertical_bounds, &horizontal_bounds);
+TEST_F(RTreeTest, NodeBuildLowBounds) {
+ ScopedVector<RTree::Node> nodes;
+ nodes.reserve(10);
+ for (int i = 1; i <= 10; ++i) {
+ nodes.push_back(new RTree::Node(Rect(0, 0, i, i), i));
+ }
+ std::vector<Rect> vertical_bounds;
+ std::vector<Rect> horizontal_bounds;
+ RTree::Node::BuildLowBounds(
+ nodes.get(), nodes.get(), &vertical_bounds, &horizontal_bounds);
for (int i = 0; i < 10; ++i) {
- EXPECT_EQ(records[i]->rect(), vertical_bounds[i]);
- EXPECT_EQ(records[i]->rect(), horizontal_bounds[i]);
+ EXPECT_EQ(vertical_bounds[i], Rect(0, 0, i + 1, i + 1));
+ EXPECT_EQ(horizontal_bounds[i], Rect(0, 0, i + 1, i + 1));
}
}
-TEST_F(RTreeNodeTest, BuildHighBounds) {
- RTreeNodes records;
- records.reserve(25);
- for (int i = 0; i < 25; ++i)
- records.push_back(new RTreeRecord(Rect(i, i, 25 - i, 25 - i), i));
-
- RTreeRects vertical_bounds;
- RTreeRects horizontal_bounds;
- NodeBuildHighBounds(
- records.get(), records.get(), &vertical_bounds, &horizontal_bounds);
+TEST_F(RTreeTest, NodeBuildHighBounds) {
+ ScopedVector<RTree::Node> nodes;
+ nodes.reserve(25);
for (int i = 0; i < 25; ++i) {
- EXPECT_EQ(records[i]->rect(), vertical_bounds[i]);
- EXPECT_EQ(records[i]->rect(), horizontal_bounds[i]);
+ nodes.push_back(new RTree::Node(Rect(i, i, 25 - i, 25 - i), i));
}
+ std::vector<Rect> vertical_bounds;
+ std::vector<Rect> horizontal_bounds;
+ RTree::Node::BuildHighBounds(
+ nodes.get(), nodes.get(), &vertical_bounds, &horizontal_bounds);
+ for (int i = 0; i < 25; ++i) {
+ EXPECT_EQ(vertical_bounds[i], Rect(i, i, 25 - i, 25 - i));
+ EXPECT_EQ(horizontal_bounds[i], Rect(i, i, 25 - i, 25 - i));
+ }
}
-TEST_F(RTreeNodeTest, ChooseSplitAxisAndIndexVertical) {
- RTreeRects low_vertical_bounds;
- RTreeRects high_vertical_bounds;
- RTreeRects low_horizontal_bounds;
- RTreeRects high_horizontal_bounds;
+TEST_F(RTreeTest, NodeChooseSplitAxisAndIndex) {
+ std::vector<Rect> low_vertical_bounds;
+ std::vector<Rect> high_vertical_bounds;
+ std::vector<Rect> low_horizontal_bounds;
+ std::vector<Rect> high_horizontal_bounds;
// In this test scenario we describe a mirrored, stacked configuration of
// horizontal, 1 pixel high rectangles labeled a-f like this:
//
@@ -426,8 +538,9 @@
// Low bounds of horizontal sort start with bounds of box a and then jump to
// cover everything, as box f is second in horizontal sort.
low_horizontal_bounds.push_back(Rect(0, 0, 5, 1));
- for (int i = 0; i < 5; ++i)
+ for (int i = 0; i < 5; ++i) {
low_horizontal_bounds.push_back(Rect(0, 0, 5, 6));
+ }
// High horizontal bounds are hand-calculated.
high_horizontal_bounds.push_back(Rect(0, 0, 5, 6));
@@ -437,24 +550,18 @@
high_horizontal_bounds.push_back(Rect(2, 2, 1, 2));
high_horizontal_bounds.push_back(Rect(2, 3, 1, 1));
- int smallest_vertical_margin =
- NodeSmallestMarginSum(2, 3, low_vertical_bounds, high_vertical_bounds);
- int smallest_horizontal_margin = NodeSmallestMarginSum(
- 2, 3, low_horizontal_bounds, high_horizontal_bounds);
- EXPECT_LT(smallest_vertical_margin, smallest_horizontal_margin);
+ // This should split vertically, right down the middle.
+ EXPECT_TRUE(RTree::Node::ChooseSplitAxis(low_vertical_bounds,
+ high_vertical_bounds,
+ low_horizontal_bounds,
+ high_horizontal_bounds,
+ 2,
+ 5));
+ EXPECT_EQ(3U,
+ RTree::Node::ChooseSplitIndex(
+ 2, 5, low_vertical_bounds, high_vertical_bounds));
- EXPECT_EQ(
- 3U,
- NodeChooseSplitIndex(2, 5, low_vertical_bounds, high_vertical_bounds));
-}
-
-TEST_F(RTreeNodeTest, ChooseSplitAxisAndIndexHorizontal) {
- RTreeRects low_vertical_bounds;
- RTreeRects high_vertical_bounds;
- RTreeRects low_horizontal_bounds;
- RTreeRects high_horizontal_bounds;
- // We rotate the shape from ChooseSplitAxisAndIndexVertical to test
- // horizontal split axis detection:
+ // We rotate the shape to test horizontal split axis detection:
//
// +--------+
// | a f |
@@ -467,11 +574,18 @@
// h sort: | 012345 |
// +--------+
//
+ // Clear out old bounds first.
+ low_vertical_bounds.clear();
+ high_vertical_bounds.clear();
+ low_horizontal_bounds.clear();
+ high_horizontal_bounds.clear();
+
// Low bounds of vertical sort start with bounds of box a and then jump to
// cover everything, as box f is second in vertical sort.
low_vertical_bounds.push_back(Rect(0, 0, 1, 5));
- for (int i = 0; i < 5; ++i)
+ for (int i = 0; i < 5; ++i) {
low_vertical_bounds.push_back(Rect(0, 0, 6, 5));
+ }
// High vertical bounds are hand-calculated.
high_vertical_bounds.push_back(Rect(0, 0, 6, 5));
@@ -489,178 +603,162 @@
high_horizontal_bounds.push_back(Rect(i, 0, 6 - i, 5));
}
- int smallest_vertical_margin =
- NodeSmallestMarginSum(2, 3, low_vertical_bounds, high_vertical_bounds);
- int smallest_horizontal_margin = NodeSmallestMarginSum(
- 2, 3, low_horizontal_bounds, high_horizontal_bounds);
-
- EXPECT_GT(smallest_vertical_margin, smallest_horizontal_margin);
-
+ // This should split horizontally, right down the middle.
+ EXPECT_FALSE(RTree::Node::ChooseSplitAxis(low_vertical_bounds,
+ high_vertical_bounds,
+ low_horizontal_bounds,
+ high_horizontal_bounds,
+ 2,
+ 5));
EXPECT_EQ(3U,
- NodeChooseSplitIndex(
+ RTree::Node::ChooseSplitIndex(
2, 5, low_horizontal_bounds, high_horizontal_bounds));
}
-TEST_F(RTreeNodeTest, DivideChildren) {
+TEST_F(RTreeTest, NodeDivideChildren) {
// Create a test node to split.
- scoped_ptr<RTreeNode> test_node(new RTreeNode);
- std::vector<RTreeNodeBase*> sorted_children;
- RTreeRects low_bounds;
- RTreeRects high_bounds;
+ scoped_ptr<RTree::Node> test_node(new RTree::Node(0));
+ std::vector<RTree::Node*> sorted_children;
+ std::vector<Rect> low_bounds;
+ std::vector<Rect> high_bounds;
// Insert 10 record nodes, also inserting them into our children array.
for (int i = 1; i <= 10; ++i) {
- scoped_ptr<RTreeRecord> record(new RTreeRecord(Rect(0, 0, i, i), i));
- sorted_children.push_back(record.get());
- test_node->AddChild(record.PassAs<RTreeNodeBase>());
+ RTree::Node* node = new RTree::Node(Rect(0, 0, i, i), i);
+ test_node->AddChild(node);
+ sorted_children.push_back(node);
low_bounds.push_back(Rect(0, 0, i, i));
high_bounds.push_back(Rect(0, 0, 10, 10));
}
// Split the children in half.
- scoped_ptr<RTreeNodeBase> split_node_base(NodeDivideChildren(
- test_node.get(), low_bounds, high_bounds, sorted_children, 5));
- RTreeNode* split_node = static_cast<RTreeNode*>(split_node_base.get());
+ scoped_ptr<RTree::Node> split_node(
+ test_node->DivideChildren(low_bounds, high_bounds, sorted_children, 5));
// Both nodes should be valid.
ValidateNode(test_node.get(), 1U, 10U);
- ValidateNode(split_node, 1U, 10U);
+ ValidateNode(split_node.get(), 1U, 10U);
// Both nodes should have five children.
- EXPECT_EQ(5U, test_node->count());
- EXPECT_EQ(5U, split_node->count());
+ EXPECT_EQ(test_node->children_.size(), 5U);
+ EXPECT_EQ(split_node->children_.size(), 5U);
// Test node should have children 1-5, split node should have children 6-10.
for (int i = 0; i < 5; ++i) {
- EXPECT_EQ(i + 1, record(test_node.get(), i)->key());
- EXPECT_EQ(i + 6, record(split_node, i)->key());
+ EXPECT_EQ(test_node->children_[i]->key(), i + 1);
+ EXPECT_EQ(split_node->children_[i]->key(), i + 6);
}
}
-TEST_F(RTreeNodeTest, RemoveChildNoOrphans) {
- scoped_ptr<RTreeNode> test_parent(new RTreeNode);
- test_parent->AddChild(
- scoped_ptr<RTreeNodeBase>(new RTreeRecord(Rect(0, 0, 1, 1), 1)));
- test_parent->AddChild(
- scoped_ptr<RTreeNodeBase>(new RTreeRecord(Rect(0, 0, 2, 2), 2)));
- test_parent->AddChild(
- scoped_ptr<RTreeNodeBase>(new RTreeRecord(Rect(0, 0, 3, 3), 3)));
+TEST_F(RTreeTest, NodeRemoveChildNoOrphans) {
+ scoped_ptr<RTree::Node> test_parent(new RTree::Node(0));
+ scoped_ptr<RTree::Node> child_one(new RTree::Node(Rect(0, 0, 1, 1), 1));
+ scoped_ptr<RTree::Node> child_two(new RTree::Node(Rect(0, 0, 2, 2), 2));
+ scoped_ptr<RTree::Node> child_three(new RTree::Node(Rect(0, 0, 3, 3), 3));
+ test_parent->AddChild(child_one.get());
+ test_parent->AddChild(child_two.get());
+ test_parent->AddChild(child_three.get());
ValidateNode(test_parent.get(), 1U, 5U);
-
- RTreeNodes orphans;
-
// Remove the middle node.
- scoped_ptr<RTreeNodeBase> middle_child(
- test_parent->RemoveChild(test_parent->child(1), &orphans));
- EXPECT_EQ(0U, orphans.size());
- EXPECT_EQ(2U, test_parent->count());
- NodeRecomputeLocalBounds(test_parent.get());
+ ScopedVector<RTree::Node> orphans;
+ EXPECT_EQ(test_parent->RemoveChild(child_two.get(), &orphans), 2U);
+ EXPECT_EQ(orphans.size(), 0U);
+ EXPECT_EQ(test_parent->count(), 2U);
+ test_parent->RecomputeBoundsNoParents();
ValidateNode(test_parent.get(), 1U, 5U);
-
// Remove the end node.
- scoped_ptr<RTreeNodeBase> end_child(
- test_parent->RemoveChild(test_parent->child(1), &orphans));
- EXPECT_EQ(0U, orphans.size());
- EXPECT_EQ(1U, test_parent->count());
- NodeRecomputeLocalBounds(test_parent.get());
+ EXPECT_EQ(test_parent->RemoveChild(child_three.get(), &orphans), 1U);
+ EXPECT_EQ(orphans.size(), 0U);
+ EXPECT_EQ(test_parent->count(), 1U);
+ test_parent->RecomputeBoundsNoParents();
ValidateNode(test_parent.get(), 1U, 5U);
-
// Remove the first node.
- scoped_ptr<RTreeNodeBase> first_child(
- test_parent->RemoveChild(test_parent->child(0), &orphans));
- EXPECT_EQ(0U, orphans.size());
- EXPECT_EQ(0U, test_parent->count());
+ EXPECT_EQ(test_parent->RemoveChild(child_one.get(), &orphans), 0U);
+ EXPECT_EQ(orphans.size(), 0U);
+ EXPECT_EQ(test_parent->count(), 0U);
}
-TEST_F(RTreeNodeTest, RemoveChildOrphans) {
- // Build binary tree of Nodes of height 4, keeping weak pointers to the
- // Levels 0 and 1 Nodes and the Records so we can test removal of them below.
- std::vector<RTreeNode*> level_1_children;
- std::vector<RTreeNode*> level_0_children;
- std::vector<RTreeRecord*> records;
- int id = 1;
- scoped_ptr<RTreeNode> root(NewNodeAtLevel(2));
- for (int i = 0; i < 2; ++i) {
- scoped_ptr<RTreeNode> level_1_child(NewNodeAtLevel(1));
- for (int j = 0; j < 2; ++j) {
- scoped_ptr<RTreeNode> level_0_child(new RTreeNode);
- for (int k = 0; k < 2; ++k) {
- scoped_ptr<RTreeRecord> record(
- new RTreeRecord(Rect(0, 0, id, id), id));
- ++id;
- records.push_back(record.get());
- level_0_child->AddChild(record.PassAs<RTreeNodeBase>());
- }
- level_0_children.push_back(level_0_child.get());
- level_1_child->AddChild(level_0_child.PassAs<RTreeNodeBase>());
- }
- level_1_children.push_back(level_1_child.get());
- root->AddChild(level_1_child.PassAs<RTreeNodeBase>());
+TEST_F(RTreeTest, NodeRemoveChildOrphans) {
+ // Build flattened binary tree of Nodes 4 deep, from the record nodes up.
+ ScopedVector<RTree::Node> nodes;
+ nodes.resize(15);
+ // Indicies 7 through 15 are record nodes.
+ for (int i = 7; i < 15; ++i) {
+ nodes[i] = new RTree::Node(Rect(0, 0, i, i), i);
}
+ // Nodes 3 through 6 are level 0 (leaves) and get 2 record nodes each.
+ for (int i = 3; i < 7; ++i) {
+ nodes[i] = new RTree::Node(0);
+ nodes[i]->AddChild(nodes[(i * 2) + 1]);
+ nodes[i]->AddChild(nodes[(i * 2) + 2]);
+ }
+ // Nodes 1 and 2 are level 1 and get 2 leaves each.
+ for (int i = 1; i < 3; ++i) {
+ nodes[i] = new RTree::Node(1);
+ nodes[i]->AddChild(nodes[(i * 2) + 1]);
+ nodes[i]->AddChild(nodes[(i * 2) + 2]);
+ }
+ // Node 0 is level 2 and gets 2 childen.
+ nodes[0] = new RTree::Node(2);
+ nodes[0]->AddChild(nodes[1]);
+ nodes[0]->AddChild(nodes[2]);
+ // This should now be a valid node structure.
+ ValidateNode(nodes[0], 2U, 2U);
- // This should now be a valid tree structure.
- ValidateNode(root.get(), 2U, 2U);
- EXPECT_EQ(2U, level_1_children.size());
- EXPECT_EQ(4U, level_0_children.size());
- EXPECT_EQ(8U, records.size());
+ // Now remove the level 0 nodes, so we get the record nodes as orphans.
+ ScopedVector<RTree::Node> orphans;
+ EXPECT_EQ(nodes[1]->RemoveChild(nodes[3], &orphans), 1U);
+ EXPECT_EQ(nodes[1]->RemoveChild(nodes[4], &orphans), 0U);
+ EXPECT_EQ(nodes[2]->RemoveChild(nodes[5], &orphans), 1U);
+ EXPECT_EQ(nodes[2]->RemoveChild(nodes[6], &orphans), 0U);
- // Now remove all of the level 0 nodes so we get the record nodes as orphans.
- RTreeNodes orphans;
- for (size_t i = 0; i < level_0_children.size(); ++i)
- level_1_children[i / 2]->RemoveChild(level_0_children[i], &orphans);
-
- // Orphans should be all 8 records but no order guarantee.
- EXPECT_EQ(8U, orphans.size());
- for (std::vector<RTreeRecord*>::iterator it = records.begin();
- it != records.end(); ++it) {
- RTreeNodes::iterator orphan =
- std::find(orphans.begin(), orphans.end(), *it);
- EXPECT_NE(orphan, orphans.end());
- orphans.erase(orphan);
+ // Orphans should be nodes 7 through 15 in order.
+ EXPECT_EQ(orphans.size(), 8U);
+ for (int i = 0; i < 8; ++i) {
+ EXPECT_EQ(orphans[i], nodes[i + 7]);
}
- EXPECT_EQ(0U, orphans.size());
+
+ // Now we remove nodes 1 and 2 from the root, expecting no further orphans.
+ // This prevents a crash due to double-delete on test exit, as no node should
+ // own any other node right now.
+ EXPECT_EQ(nodes[0]->RemoveChild(nodes[1], &orphans), 1U);
+ EXPECT_EQ(orphans.size(), 8U);
+ EXPECT_EQ(nodes[0]->RemoveChild(nodes[2], &orphans), 0U);
+ EXPECT_EQ(orphans.size(), 8U);
+
+ // Prevent double-delete of nodes by both nodes and orphans.
+ orphans.weak_clear();
}
-TEST_F(RTreeNodeTest, RemoveAndReturnLastChild) {
- scoped_ptr<RTreeNode> test_parent(new RTreeNode);
- test_parent->AddChild(
- scoped_ptr<RTreeNodeBase>(new RTreeRecord(Rect(0, 0, 1, 1), 1)));
- test_parent->AddChild(
- scoped_ptr<RTreeNodeBase>(new RTreeRecord(Rect(0, 0, 2, 2), 2)));
- test_parent->AddChild(
- scoped_ptr<RTreeNodeBase>(new RTreeRecord(Rect(0, 0, 3, 3), 3)));
+TEST_F(RTreeTest, NodeRemoveAndReturnLastChild) {
+ scoped_ptr<RTree::Node> test_parent(new RTree::Node(0));
+ scoped_ptr<RTree::Node> child_one(new RTree::Node(Rect(0, 0, 1, 1), 1));
+ scoped_ptr<RTree::Node> child_two(new RTree::Node(Rect(0, 0, 2, 2), 2));
+ scoped_ptr<RTree::Node> child_three(new RTree::Node(Rect(0, 0, 3, 3), 3));
+ test_parent->AddChild(child_one.get());
+ test_parent->AddChild(child_two.get());
+ test_parent->AddChild(child_three.get());
ValidateNode(test_parent.get(), 1U, 5U);
- RTreeNodeBase* child = test_parent->child(2);
- scoped_ptr<RTreeNodeBase> last_child(test_parent->RemoveAndReturnLastChild());
- EXPECT_EQ(child, last_child.get());
- EXPECT_EQ(2U, test_parent->count());
- NodeRecomputeLocalBounds(test_parent.get());
+ EXPECT_EQ(test_parent->RemoveAndReturnLastChild().release(),
+ child_three.get());
+ EXPECT_EQ(test_parent->count(), 2U);
+ test_parent->RecomputeBoundsNoParents();
ValidateNode(test_parent.get(), 1U, 5U);
- child = test_parent->child(1);
- scoped_ptr<RTreeNodeBase> middle_child(
- test_parent->RemoveAndReturnLastChild());
- EXPECT_EQ(child, middle_child.get());
- EXPECT_EQ(1U, test_parent->count());
- NodeRecomputeLocalBounds(test_parent.get());
+ EXPECT_EQ(test_parent->RemoveAndReturnLastChild().release(), child_two.get());
+ EXPECT_EQ(test_parent->count(), 1U);
+ test_parent->RecomputeBoundsNoParents();
ValidateNode(test_parent.get(), 1U, 5U);
- child = test_parent->child(0);
- scoped_ptr<RTreeNodeBase> first_child(
- test_parent->RemoveAndReturnLastChild());
- EXPECT_EQ(child, first_child.get());
- EXPECT_EQ(0U, test_parent->count());
+ EXPECT_EQ(test_parent->RemoveAndReturnLastChild().release(), child_one.get());
+ EXPECT_EQ(test_parent->count(), 0U);
}
-TEST_F(RTreeNodeTest, LeastOverlapIncrease) {
- scoped_ptr<RTreeNode> test_parent(NewNodeAtLevel(1));
- // Construct 4 nodes with 1x2 rects spaced horizontally 1 pixel apart, or:
+TEST_F(RTreeTest, NodeLeastOverlapIncrease) {
+ scoped_ptr<RTree::Node> test_parent(new RTree::Node(0));
+ // Construct 4 nodes with 1x2 retangles spaced horizontally 1 pixel apart, or:
//
// a b c d
// a b c d
//
for (int i = 0; i < 4; ++i) {
- scoped_ptr<RTreeNode> node(new RTreeNode);
- scoped_ptr<RTreeRecord> record(
- new RTreeRecord(Rect(i * 2, 0, 1, 2), i + 1));
- node->AddChild(record.PassAs<RTreeNodeBase>());
- test_parent->AddChild(node.PassAs<RTreeNodeBase>());
+ test_parent->AddChild(new RTree::Node(Rect(i * 2, 0, 1, 2), i + 1));
}
ValidateNode(test_parent.get(), 1U, 5U);
@@ -672,11 +770,11 @@
// a b c d
//
Rect test_rect_far(7, 0, 1, 1);
- RTreeRects expanded_rects;
+ std::vector<Rect> expanded_rects;
BuildExpandedRects(test_parent.get(), test_rect_far, &expanded_rects);
- RTreeNode* result = NodeLeastOverlapIncrease(
- test_parent.get(), test_rect_far, expanded_rects);
- EXPECT_EQ(4, record(result, 0)->key());
+ RTree::Node* result =
+ test_parent->LeastOverlapIncrease(test_rect_far, expanded_rects);
+ EXPECT_EQ(result->key(), 4);
// Test rect covering the bottom half of all children should be a 4-way tie,
// so LeastOverlapIncrease should return NULL:
@@ -686,8 +784,7 @@
//
Rect test_rect_tie(0, 1, 7, 1);
BuildExpandedRects(test_parent.get(), test_rect_tie, &expanded_rects);
- result = NodeLeastOverlapIncrease(
- test_parent.get(), test_rect_tie, expanded_rects);
+ result = test_parent->LeastOverlapIncrease(test_rect_tie, expanded_rects);
EXPECT_TRUE(result == NULL);
// Test rect completely inside c should return the third rectangle:
@@ -697,9 +794,8 @@
//
Rect test_rect_inside(4, 0, 1, 1);
BuildExpandedRects(test_parent.get(), test_rect_inside, &expanded_rects);
- result = NodeLeastOverlapIncrease(
- test_parent.get(), test_rect_inside, expanded_rects);
- EXPECT_EQ(3, record(result, 0)->key());
+ result = test_parent->LeastOverlapIncrease(test_rect_inside, expanded_rects);
+ EXPECT_EQ(result->key(), 3);
// Add a rectangle that overlaps completely with rectangle c, to test
// when there is a tie between two completely contained rectangles:
@@ -707,40 +803,24 @@
// a b Ted
// a b eed
//
- scoped_ptr<RTreeNode> record_parent(new RTreeNode);
- record_parent->AddChild(
- scoped_ptr<RTreeNodeBase>(new RTreeRecord(Rect(4, 0, 2, 2), 9)));
- test_parent->AddChild(record_parent.PassAs<RTreeNodeBase>());
+ test_parent->AddChild(new RTree::Node(Rect(4, 0, 2, 2), 9));
BuildExpandedRects(test_parent.get(), test_rect_inside, &expanded_rects);
- result = NodeLeastOverlapIncrease(
- test_parent.get(), test_rect_inside, expanded_rects);
+ result = test_parent->LeastOverlapIncrease(test_rect_inside, expanded_rects);
EXPECT_TRUE(result == NULL);
}
-TEST_F(RTreeNodeTest, LeastAreaEnlargement) {
- scoped_ptr<RTreeNode> test_parent(NewNodeAtLevel(1));
+TEST_F(RTreeTest, NodeLeastAreaEnlargement) {
+ scoped_ptr<RTree::Node> test_parent(new RTree::Node(0));
// Construct 4 nodes in a cross-hairs style configuration:
//
// a
// b c
// d
//
- scoped_ptr<RTreeNode> node(new RTreeNode);
- node->AddChild(
- scoped_ptr<RTreeNodeBase>(new RTreeRecord(Rect(1, 0, 1, 1), 1)));
- test_parent->AddChild(node.PassAs<RTreeNodeBase>());
- node.reset(new RTreeNode);
- node->AddChild(
- scoped_ptr<RTreeNodeBase>(new RTreeRecord(Rect(0, 1, 1, 1), 2)));
- test_parent->AddChild(node.PassAs<RTreeNodeBase>());
- node.reset(new RTreeNode);
- node->AddChild(
- scoped_ptr<RTreeNodeBase>(new RTreeRecord(Rect(2, 1, 1, 1), 3)));
- test_parent->AddChild(node.PassAs<RTreeNodeBase>());
- node.reset(new RTreeNode);
- node->AddChild(
- scoped_ptr<RTreeNodeBase>(new RTreeRecord(Rect(1, 2, 1, 1), 4)));
- test_parent->AddChild(node.PassAs<RTreeNodeBase>());
+ test_parent->AddChild(new RTree::Node(Rect(1, 0, 1, 1), 1));
+ test_parent->AddChild(new RTree::Node(Rect(0, 1, 1, 1), 2));
+ test_parent->AddChild(new RTree::Node(Rect(2, 1, 1, 1), 3));
+ test_parent->AddChild(new RTree::Node(Rect(1, 2, 1, 1), 4));
ValidateNode(test_parent.get(), 1U, 5U);
@@ -752,11 +832,11 @@
// T
//
Rect test_rect_below(1, 3, 1, 1);
- RTreeRects expanded_rects;
+ std::vector<Rect> expanded_rects;
BuildExpandedRects(test_parent.get(), test_rect_below, &expanded_rects);
- RTreeNode* result = NodeLeastAreaEnlargement(
- test_parent.get(), test_rect_below, expanded_rects);
- EXPECT_EQ(4, record(result, 0)->key());
+ RTree::Node* result =
+ test_parent->LeastAreaEnlargement(test_rect_below, expanded_rects);
+ EXPECT_EQ(result->key(), 4);
// Test rect completely inside b should require minimum area to add to Node b:
//
@@ -766,9 +846,8 @@
//
Rect test_rect_inside(0, 1, 1, 1);
BuildExpandedRects(test_parent.get(), test_rect_inside, &expanded_rects);
- result = NodeLeastAreaEnlargement(
- test_parent.get(), test_rect_inside, expanded_rects);
- EXPECT_EQ(2, record(result, 0)->key());
+ result = test_parent->LeastAreaEnlargement(test_rect_inside, expanded_rects);
+ EXPECT_EQ(result->key(), 2);
// Add e at (0, 1) to overlap b and c, to test tie-breaking:
//
@@ -776,10 +855,7 @@
// eee
// d
//
- node.reset(new RTreeNode);
- node->AddChild(
- scoped_ptr<RTreeNodeBase>(new RTreeRecord(Rect(0, 1, 3, 1), 7)));
- test_parent->AddChild(node.PassAs<RTreeNodeBase>());
+ test_parent->AddChild(new RTree::Node(Rect(0, 1, 3, 1), 7));
ValidateNode(test_parent.get(), 1U, 5U);
@@ -793,222 +869,9 @@
//
Rect test_rect_tie_breaker(3, 1, 1, 1);
BuildExpandedRects(test_parent.get(), test_rect_tie_breaker, &expanded_rects);
- result = NodeLeastAreaEnlargement(
- test_parent.get(), test_rect_tie_breaker, expanded_rects);
- EXPECT_EQ(3, record(result, 0)->key());
+ result =
+ test_parent->LeastAreaEnlargement(test_rect_tie_breaker, expanded_rects);
+ EXPECT_EQ(result->key(), 3);
}
-// RTreeTest ------------------------------------------------------------------
-
-// An empty RTree should never return AppendIntersectingRecords results, and
-// RTrees should be empty upon construction.
-TEST_F(RTreeTest, AppendIntersectingRecordsOnEmptyTree) {
- RT rt(2, 10);
- ValidateRTree(&rt);
- RT::Matches results;
- Rect test_rect(25, 25);
- rt.AppendIntersectingRecords(test_rect, &results);
- EXPECT_EQ(0U, results.size());
- ValidateRTree(&rt);
-}
-
-// Clear should empty the tree, meaning that all queries should not return
-// results after.
-TEST_F(RTreeTest, ClearEmptiesTreeOfSingleNode) {
- RT rt(2, 5);
- rt.Insert(Rect(0, 0, 100, 100), 1);
- rt.Clear();
- RT::Matches results;
- Rect test_rect(1, 1);
- rt.AppendIntersectingRecords(test_rect, &results);
- EXPECT_EQ(0U, results.size());
- ValidateRTree(&rt);
-}
-
-// Even with a complex internal structure, clear should empty the tree, meaning
-// that all queries should not return results after.
-TEST_F(RTreeTest, ClearEmptiesTreeOfManyNodes) {
- RT rt(2, 5);
- AddStackedSquares(&rt, 100);
- rt.Clear();
- RT::Matches results;
- Rect test_rect(1, 1);
- rt.AppendIntersectingRecords(test_rect, &results);
- EXPECT_EQ(0U, results.size());
- ValidateRTree(&rt);
-}
-
-// Duplicate inserts should overwrite previous inserts.
-TEST_F(RTreeTest, DuplicateInsertsOverwrite) {
- RT rt(2, 5);
- // Add 100 stacked squares, but always with duplicate key of 0.
- for (int i = 1; i <= 100; ++i) {
- rt.Insert(Rect(0, 0, i, i), 0);
- ValidateRTree(&rt);
- }
- RT::Matches results;
- Rect test_rect(1, 1);
- rt.AppendIntersectingRecords(test_rect, &results);
- EXPECT_EQ(1U, results.size());
- EXPECT_EQ(1U, results.count(0));
-}
-
-// Call Remove() once on something that's been inserted repeatedly.
-TEST_F(RTreeTest, DuplicateInsertRemove) {
- RT rt(3, 9);
- AddStackedSquares(&rt, 25);
- for (int i = 1; i <= 100; ++i) {
- rt.Insert(Rect(0, 0, i, i), 26);
- ValidateRTree(&rt);
- }
- rt.Remove(26);
- RT::Matches results;
- Rect test_rect(1, 1);
- rt.AppendIntersectingRecords(test_rect, &results);
- EXPECT_EQ(25U, results.size());
- VerifyAllKeys(results);
-}
-
-// Call Remove() repeatedly on something that's been inserted once.
-TEST_F(RTreeTest, InsertDuplicateRemove) {
- RT rt(7, 15);
- AddStackedSquares(&rt, 101);
- for (int i = 0; i < 100; ++i) {
- rt.Remove(101);
- ValidateRTree(&rt);
- }
- RT::Matches results;
- Rect test_rect(1, 1);
- rt.AppendIntersectingRecords(test_rect, &results);
- EXPECT_EQ(100U, results.size());
- VerifyAllKeys(results);
-}
-
-// Stacked rects should meet all matching queries regardless of nesting.
-TEST_F(RTreeTest, AppendIntersectingRecordsStackedSquaresNestedHit) {
- RT rt(2, 5);
- AddStackedSquares(&rt, 100);
- RT::Matches results;
- Rect test_rect(1, 1);
- rt.AppendIntersectingRecords(test_rect, &results);
- EXPECT_EQ(100U, results.size());
- VerifyAllKeys(results);
-}
-
-// Stacked rects should meet all matching queries when contained completely by
-// the query rectangle.
-TEST_F(RTreeTest, AppendIntersectingRecordsStackedSquaresContainedHit) {
- RT rt(2, 10);
- AddStackedSquares(&rt, 100);
- RT::Matches results;
- Rect test_rect(0, 0, 100, 100);
- rt.AppendIntersectingRecords(test_rect, &results);
- EXPECT_EQ(100U, results.size());
- VerifyAllKeys(results);
-}
-
-// Stacked rects should miss a missing query when the query has no intersection
-// with the rects.
-TEST_F(RTreeTest, AppendIntersectingRecordsStackedSquaresCompleteMiss) {
- RT rt(2, 7);
- AddStackedSquares(&rt, 100);
- RT::Matches results;
- Rect test_rect(150, 150, 100, 100);
- rt.AppendIntersectingRecords(test_rect, &results);
- EXPECT_EQ(0U, results.size());
-}
-
-// Removing half the nodes after insertion should still result in a valid tree.
-TEST_F(RTreeTest, RemoveHalfStackedRects) {
- RT rt(2, 11);
- AddStackedSquares(&rt, 200);
- for (int i = 101; i <= 200; ++i) {
- rt.Remove(i);
- ValidateRTree(&rt);
- }
- RT::Matches results;
- Rect test_rect(1, 1);
- rt.AppendIntersectingRecords(test_rect, &results);
- EXPECT_EQ(100U, results.size());
- VerifyAllKeys(results);
-
- // Add the nodes back in.
- for (int i = 101; i <= 200; ++i) {
- rt.Insert(Rect(0, 0, i, i), i);
- ValidateRTree(&rt);
- }
- results.clear();
- rt.AppendIntersectingRecords(test_rect, &results);
- EXPECT_EQ(200U, results.size());
- VerifyAllKeys(results);
-}
-
-TEST_F(RTreeTest, InsertDupToRoot) {
- RT rt(2, 5);
- rt.Insert(Rect(0, 0, 1, 2), 1);
- ValidateRTree(&rt);
- rt.Insert(Rect(0, 0, 2, 1), 1);
- ValidateRTree(&rt);
-}
-
-TEST_F(RTreeTest, InsertNegativeCoordsRect) {
- RT rt(5, 11);
- for (int i = 1; i <= 100; ++i) {
- rt.Insert(Rect(-i, -i, i, i), (i * 2) - 1);
- ValidateRTree(&rt);
- rt.Insert(Rect(0, 0, i, i), i * 2);
- ValidateRTree(&rt);
- }
- RT::Matches results;
- Rect test_rect(-1, -1, 2, 2);
- rt.AppendIntersectingRecords(test_rect, &results);
- EXPECT_EQ(200U, results.size());
- VerifyAllKeys(results);
-}
-
-TEST_F(RTreeTest, RemoveNegativeCoordsRect) {
- RT rt(7, 21);
-
- // Add 100 positive stacked squares.
- AddStackedSquares(&rt, 100);
-
- // Now add 100 negative stacked squares.
- for (int i = 101; i <= 200; ++i) {
- rt.Insert(Rect(100 - i, 100 - i, i - 100, i - 100), 301 - i);
- ValidateRTree(&rt);
- }
-
- // Now remove half of the negative squares.
- for (int i = 101; i <= 150; ++i) {
- rt.Remove(301 - i);
- ValidateRTree(&rt);
- }
-
- // Queries should return 100 positive and 50 negative stacked squares.
- RT::Matches results;
- Rect test_rect(-1, -1, 2, 2);
- rt.AppendIntersectingRecords(test_rect, &results);
- EXPECT_EQ(150U, results.size());
- VerifyAllKeys(results);
-}
-
-TEST_F(RTreeTest, InsertEmptyRectReplacementRemovesKey) {
- RT rt(10, 31);
- AddStackedSquares(&rt, 50);
- ValidateRTree(&rt);
-
- // Replace last square with empty rect.
- rt.Insert(Rect(), 50);
- ValidateRTree(&rt);
-
- // Now query large area to get all rects in tree.
- RT::Matches results;
- Rect test_rect(0, 0, 100, 100);
- rt.AppendIntersectingRecords(test_rect, &results);
-
- // Should only be 49 rects in tree.
- EXPECT_EQ(49U, results.size());
- VerifyAllKeys(results);
-}
-
} // namespace gfx
« no previous file with comments | « trunk/src/ui/gfx/geometry/r_tree_base.cc ('k') | trunk/src/ui/gfx/gfx.gyp » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698