|  | // Copyright 2013 The Flutter Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "rtree.h" | 
|  |  | 
|  | #include <list> | 
|  |  | 
|  | #include "flutter/fml/logging.h" | 
|  | #include "third_party/skia/include/core/SkBBHFactory.h" | 
|  |  | 
|  | namespace flutter { | 
|  |  | 
|  | RTree::RTree() : bbh_{SkRTreeFactory{}()}, all_ops_count_(0) {} | 
|  |  | 
|  | void RTree::insert(const SkRect boundsArray[], | 
|  | const SkBBoxHierarchy::Metadata metadata[], | 
|  | int N) { | 
|  | FML_DCHECK(0 == all_ops_count_); | 
|  | bbh_->insert(boundsArray, metadata, N); | 
|  | for (int i = 0; i < N; i++) { | 
|  | if (metadata != nullptr && metadata[i].isDraw) { | 
|  | draw_op_[i] = boundsArray[i]; | 
|  | } | 
|  | } | 
|  | all_ops_count_ = N; | 
|  | } | 
|  |  | 
|  | void RTree::insert(const SkRect boundsArray[], int N) { | 
|  | insert(boundsArray, nullptr, N); | 
|  | } | 
|  |  | 
|  | void RTree::search(const SkRect& query, std::vector<int>* results) const { | 
|  | bbh_->search(query, results); | 
|  | } | 
|  |  | 
|  | std::list<SkRect> RTree::searchNonOverlappingDrawnRects( | 
|  | const SkRect& query) const { | 
|  | // Get the indexes for the operations that intersect with the query rect. | 
|  | std::vector<int> intermediary_results; | 
|  | search(query, &intermediary_results); | 
|  |  | 
|  | std::list<SkRect> final_results; | 
|  | for (int index : intermediary_results) { | 
|  | auto draw_op = draw_op_.find(index); | 
|  | // Ignore records that don't draw anything. | 
|  | if (draw_op == draw_op_.end()) { | 
|  | continue; | 
|  | } | 
|  | auto current_record_rect = draw_op->second; | 
|  | auto replaced_existing_rect = false; | 
|  | // // If the current record rect intersects with any of the rects in the | 
|  | // // result list, then join them, and update the rect in final_results. | 
|  | std::list<SkRect>::iterator curr_rect_itr = final_results.begin(); | 
|  | std::list<SkRect>::iterator first_intersecting_rect_itr; | 
|  | while (!replaced_existing_rect && curr_rect_itr != final_results.end()) { | 
|  | if (SkRect::Intersects(*curr_rect_itr, current_record_rect)) { | 
|  | replaced_existing_rect = true; | 
|  | first_intersecting_rect_itr = curr_rect_itr; | 
|  | curr_rect_itr->join(current_record_rect); | 
|  | } | 
|  | curr_rect_itr++; | 
|  | } | 
|  | // It's possible that the result contains duplicated rects at this point. | 
|  | // For example, consider a result list that contains rects A, B. If a | 
|  | // new rect C is a superset of A and B, then A and B are the same set after | 
|  | // the merge. As a result, find such cases and remove them from the result | 
|  | // list. | 
|  | while (replaced_existing_rect && curr_rect_itr != final_results.end()) { | 
|  | if (SkRect::Intersects(*curr_rect_itr, *first_intersecting_rect_itr)) { | 
|  | first_intersecting_rect_itr->join(*curr_rect_itr); | 
|  | curr_rect_itr = final_results.erase(curr_rect_itr); | 
|  | } else { | 
|  | curr_rect_itr++; | 
|  | } | 
|  | } | 
|  | if (!replaced_existing_rect) { | 
|  | final_results.push_back(current_record_rect); | 
|  | } | 
|  | } | 
|  | return final_results; | 
|  | } | 
|  |  | 
|  | size_t RTree::bytesUsed() const { | 
|  | return bbh_->bytesUsed(); | 
|  | } | 
|  |  | 
|  | RTreeFactory::RTreeFactory() { | 
|  | r_tree_ = sk_make_sp<RTree>(); | 
|  | } | 
|  |  | 
|  | sk_sp<RTree> RTreeFactory::getInstance() { | 
|  | return r_tree_; | 
|  | } | 
|  |  | 
|  | sk_sp<SkBBoxHierarchy> RTreeFactory::operator()() const { | 
|  | return r_tree_; | 
|  | } | 
|  |  | 
|  | }  // namespace flutter |