xref: /aosp_15_r20/external/skia/modules/bentleyottmann/src/EventQueue.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker // Copyright 2023 Google LLC
2*c8dee2aaSAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be found in the LICENSE file.
3*c8dee2aaSAndroid Build Coastguard Worker 
4*c8dee2aaSAndroid Build Coastguard Worker #include "modules/bentleyottmann/include/EventQueue.h"
5*c8dee2aaSAndroid Build Coastguard Worker 
6*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkAssert.h"
7*c8dee2aaSAndroid Build Coastguard Worker #include <algorithm>
8*c8dee2aaSAndroid Build Coastguard Worker #include <cstdint>
9*c8dee2aaSAndroid Build Coastguard Worker #include <utility>
10*c8dee2aaSAndroid Build Coastguard Worker 
11*c8dee2aaSAndroid Build Coastguard Worker namespace bentleyottmann {
12*c8dee2aaSAndroid Build Coastguard Worker 
13*c8dee2aaSAndroid Build Coastguard Worker // -- EventQueue -----------------------------------------------------------------------------------
Make(SkSpan<const Segment> segments)14*c8dee2aaSAndroid Build Coastguard Worker std::optional<EventQueue> EventQueue::Make(SkSpan<const Segment> segments) {
15*c8dee2aaSAndroid Build Coastguard Worker     Queue queue;
16*c8dee2aaSAndroid Build Coastguard Worker 
17*c8dee2aaSAndroid Build Coastguard Worker     int32_t left   = Point::Largest().x,
18*c8dee2aaSAndroid Build Coastguard Worker             top    = Point::Largest().y,
19*c8dee2aaSAndroid Build Coastguard Worker             right  = Point::Smallest().x,
20*c8dee2aaSAndroid Build Coastguard Worker             bottom = Point::Smallest().y;
21*c8dee2aaSAndroid Build Coastguard Worker 
22*c8dee2aaSAndroid Build Coastguard Worker     for(const Segment& s : segments) {
23*c8dee2aaSAndroid Build Coastguard Worker         auto [l, t, r, b] = s.bounds();
24*c8dee2aaSAndroid Build Coastguard Worker         left   = std::min(l, left);
25*c8dee2aaSAndroid Build Coastguard Worker         top    = std::min(t, top);
26*c8dee2aaSAndroid Build Coastguard Worker         right  = std::max(r, right);
27*c8dee2aaSAndroid Build Coastguard Worker         bottom = std::max(b, bottom);
28*c8dee2aaSAndroid Build Coastguard Worker 
29*c8dee2aaSAndroid Build Coastguard Worker         queue.insert(Event{s.upper(), Upper{s}});
30*c8dee2aaSAndroid Build Coastguard Worker     }
31*c8dee2aaSAndroid Build Coastguard Worker 
32*c8dee2aaSAndroid Build Coastguard Worker     // If min max difference is too large, fail.
33*c8dee2aaSAndroid Build Coastguard Worker     if (Point::DifferenceTooBig(Point{left, top}, Point{right, bottom})) {
34*c8dee2aaSAndroid Build Coastguard Worker         return std::nullopt;
35*c8dee2aaSAndroid Build Coastguard Worker     }
36*c8dee2aaSAndroid Build Coastguard Worker 
37*c8dee2aaSAndroid Build Coastguard Worker     return EventQueue{std::move(queue)};
38*c8dee2aaSAndroid Build Coastguard Worker }
39*c8dee2aaSAndroid Build Coastguard Worker 
EventQueue(EventQueue::Queue && queue)40*c8dee2aaSAndroid Build Coastguard Worker EventQueue::EventQueue(EventQueue::Queue&& queue) : fQueue{std::move(queue)} { }
41*c8dee2aaSAndroid Build Coastguard Worker 
add(const Event & event)42*c8dee2aaSAndroid Build Coastguard Worker void EventQueue::add(const Event& event) {
43*c8dee2aaSAndroid Build Coastguard Worker     // New events must be up stream from the current event.
44*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(fLastEventPoint < event.where);
45*c8dee2aaSAndroid Build Coastguard Worker 
46*c8dee2aaSAndroid Build Coastguard Worker     fQueue.insert(event);
47*c8dee2aaSAndroid Build Coastguard Worker }
48*c8dee2aaSAndroid Build Coastguard Worker 
addCrossing(Point crossingPoint,const Segment & s0,const Segment & s1)49*c8dee2aaSAndroid Build Coastguard Worker void EventQueue::addCrossing(Point crossingPoint, const Segment& s0, const Segment& s1) {
50*c8dee2aaSAndroid Build Coastguard Worker     this->add({crossingPoint, Cross{s0, s1}});
51*c8dee2aaSAndroid Build Coastguard Worker     fCrossings.push_back({s0, s1, crossingPoint});
52*c8dee2aaSAndroid Build Coastguard Worker }
53*c8dee2aaSAndroid Build Coastguard Worker 
hasMoreEvents() const54*c8dee2aaSAndroid Build Coastguard Worker bool EventQueue::hasMoreEvents() const {
55*c8dee2aaSAndroid Build Coastguard Worker     return !fQueue.empty();
56*c8dee2aaSAndroid Build Coastguard Worker }
57*c8dee2aaSAndroid Build Coastguard Worker 
58*c8dee2aaSAndroid Build Coastguard Worker template<class... Ts>
59*c8dee2aaSAndroid Build Coastguard Worker struct Visitor : Ts... { using Ts::operator()...; };
60*c8dee2aaSAndroid Build Coastguard Worker template<class... Ts>
61*c8dee2aaSAndroid Build Coastguard Worker Visitor(Ts...) -> Visitor<Ts...>;
62*c8dee2aaSAndroid Build Coastguard Worker 
handleNextEventPoint(SweepLineInterface * handler)63*c8dee2aaSAndroid Build Coastguard Worker void EventQueue::handleNextEventPoint(SweepLineInterface* handler) {
64*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(!fQueue.empty());
65*c8dee2aaSAndroid Build Coastguard Worker 
66*c8dee2aaSAndroid Build Coastguard Worker     // Clear temp segment buffers.
67*c8dee2aaSAndroid Build Coastguard Worker     fDeletionSet.clear();
68*c8dee2aaSAndroid Build Coastguard Worker     fInsertionSet.clear();
69*c8dee2aaSAndroid Build Coastguard Worker 
70*c8dee2aaSAndroid Build Coastguard Worker     // An events that are Lower points.
71*c8dee2aaSAndroid Build Coastguard Worker     bool hasLower = false;
72*c8dee2aaSAndroid Build Coastguard Worker 
73*c8dee2aaSAndroid Build Coastguard Worker     // Set up the visitors for the different event types.
74*c8dee2aaSAndroid Build Coastguard Worker     auto handleLower = [&hasLower](const Lower& l) {
75*c8dee2aaSAndroid Build Coastguard Worker         hasLower = true;
76*c8dee2aaSAndroid Build Coastguard Worker     };
77*c8dee2aaSAndroid Build Coastguard Worker 
78*c8dee2aaSAndroid Build Coastguard Worker     // Crossing Segments must be deleted and re-inserted in the sweep line.
79*c8dee2aaSAndroid Build Coastguard Worker     auto handleCross = [this](const Cross& c) {
80*c8dee2aaSAndroid Build Coastguard Worker         fDeletionSet.insert({c.s0, c.s1});
81*c8dee2aaSAndroid Build Coastguard Worker         fInsertionSet.insert({c.s0, c.s1});
82*c8dee2aaSAndroid Build Coastguard Worker     };
83*c8dee2aaSAndroid Build Coastguard Worker 
84*c8dee2aaSAndroid Build Coastguard Worker     // Upper events are added to the sweep line, and a lower event is added to the event queue.
85*c8dee2aaSAndroid Build Coastguard Worker     auto handleUpper = [this](const Upper& u) {
86*c8dee2aaSAndroid Build Coastguard Worker         fInsertionSet.insert(u.s);
87*c8dee2aaSAndroid Build Coastguard Worker         // Add the delete event for the inserted segment. Make sure we are not adding more events
88*c8dee2aaSAndroid Build Coastguard Worker         // on this eventPoint.
89*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(u.s.lower() != u.s.upper());
90*c8dee2aaSAndroid Build Coastguard Worker         this->add(Event{u.s.lower(), Lower{}});
91*c8dee2aaSAndroid Build Coastguard Worker     };
92*c8dee2aaSAndroid Build Coastguard Worker 
93*c8dee2aaSAndroid Build Coastguard Worker     Visitor visitor{handleLower, handleCross, handleUpper};
94*c8dee2aaSAndroid Build Coastguard Worker 
95*c8dee2aaSAndroid Build Coastguard Worker     const Point eventPoint = fQueue.begin()->where;
96*c8dee2aaSAndroid Build Coastguard Worker 
97*c8dee2aaSAndroid Build Coastguard Worker     // We must make forward progress.
98*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(fLastEventPoint < eventPoint);
99*c8dee2aaSAndroid Build Coastguard Worker     fLastEventPoint = eventPoint;
100*c8dee2aaSAndroid Build Coastguard Worker 
101*c8dee2aaSAndroid Build Coastguard Worker     // Accumulate changes for all events with the same event point.
102*c8dee2aaSAndroid Build Coastguard Worker     auto cursor = fQueue.begin();
103*c8dee2aaSAndroid Build Coastguard Worker     const auto queueEnd = fQueue.end();
104*c8dee2aaSAndroid Build Coastguard Worker     for (; cursor != queueEnd && cursor->where == eventPoint;
105*c8dee2aaSAndroid Build Coastguard Worker          ++cursor) {
106*c8dee2aaSAndroid Build Coastguard Worker         const Event& event = *cursor;
107*c8dee2aaSAndroid Build Coastguard Worker         std::visit(visitor, event.type);
108*c8dee2aaSAndroid Build Coastguard Worker     }
109*c8dee2aaSAndroid Build Coastguard Worker 
110*c8dee2aaSAndroid Build Coastguard Worker     // Remove all accumulated events with the same event point.
111*c8dee2aaSAndroid Build Coastguard Worker     fQueue.erase(fQueue.begin(), cursor);
112*c8dee2aaSAndroid Build Coastguard Worker 
113*c8dee2aaSAndroid Build Coastguard Worker     if (hasLower || !fDeletionSet.empty()) {
114*c8dee2aaSAndroid Build Coastguard Worker         // There are segments to delete.
115*c8dee2aaSAndroid Build Coastguard Worker         handler->handleDeletions(eventPoint, fDeletionSet);
116*c8dee2aaSAndroid Build Coastguard Worker     }
117*c8dee2aaSAndroid Build Coastguard Worker 
118*c8dee2aaSAndroid Build Coastguard Worker     if (hasLower || !fDeletionSet.empty() || !fInsertionSet.empty()) {
119*c8dee2aaSAndroid Build Coastguard Worker         // If there are insertions then insert them. If there are no insertions, but there were
120*c8dee2aaSAndroid Build Coastguard Worker         // deletions we need to check for new crossings.
121*c8dee2aaSAndroid Build Coastguard Worker         handler->handleInsertionsAndCheckForNewCrossings(eventPoint, fInsertionSet, this);
122*c8dee2aaSAndroid Build Coastguard Worker     }
123*c8dee2aaSAndroid Build Coastguard Worker }
124*c8dee2aaSAndroid Build Coastguard Worker 
crossings()125*c8dee2aaSAndroid Build Coastguard Worker std::vector<Crossing> EventQueue::crossings() {
126*c8dee2aaSAndroid Build Coastguard Worker     return std::vector<Crossing>{fCrossings.begin(), fCrossings.end()};
127*c8dee2aaSAndroid Build Coastguard Worker }
128*c8dee2aaSAndroid Build Coastguard Worker 
operator ()(const bentleyottmann::Segment & s0,const bentleyottmann::Segment & s1) const129*c8dee2aaSAndroid Build Coastguard Worker bool OrderBySlope::operator()(const bentleyottmann::Segment& s0,
130*c8dee2aaSAndroid Build Coastguard Worker                               const bentleyottmann::Segment& s1) const {
131*c8dee2aaSAndroid Build Coastguard Worker     return compare_slopes(s0, s1) < 0;
132*c8dee2aaSAndroid Build Coastguard Worker }
133*c8dee2aaSAndroid Build Coastguard Worker }  // namespace bentleyottmann
134