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