summaryrefslogtreecommitdiff
path: root/src/core
diff options
context:
space:
mode:
authorPatric Stout <truebrain@openttd.org>2021-02-17 21:19:32 +0100
committerGitHub <noreply@github.com>2021-02-17 21:19:32 +0100
commit8e0d48a0f6b2900da9561126af73a046afb58d77 (patch)
treeb9142771d13c32b7d428259dd82fb943ed2391c2 /src/core
parent8bd2349ab6d25a16b8ffc4e9b4ea541ca8d29610 (diff)
downloadopenttd-8e0d48a0f6b2900da9561126af73a046afb58d77.tar.xz
Fix: [SDL2] simplify what to redraw to prevent tearing (#8685)
When there are a lot of rects to redraw, of which one of the last ones is almost the full screen, visual tearing happens over the vertical axis. This is most visible when scrolling the map. This can be prevented by using less rects. To simplify the situation, and as solutions like OpenGL need this anyway, keep a single rect that shows the biggest size that updates everything correctly. Although this means it needs a bit more time redrawing where it is strictly seen not needed, it also means less commands have to be executed in the backend. In the end, this is a trade-off, and from experiments it seems the approach of this commit gives a better result.
Diffstat (limited to 'src/core')
-rw-r--r--src/core/geometry_func.cpp22
-rw-r--r--src/core/geometry_func.hpp12
2 files changed, 34 insertions, 0 deletions
diff --git a/src/core/geometry_func.cpp b/src/core/geometry_func.cpp
index 89e70fb5b..dccef72ec 100644
--- a/src/core/geometry_func.cpp
+++ b/src/core/geometry_func.cpp
@@ -26,3 +26,25 @@ Dimension maxdim(const Dimension &d1, const Dimension &d2)
d.height = std::max(d1.height, d2.height);
return d;
}
+
+/**
+ * Compute the bounding rectangle around two rectangles.
+ * @param r1 First rectangle.
+ * @param r2 Second rectangle.
+ * @return The bounding rectangle, the smallest rectangle that contains both arguments.
+ */
+Rect BoundingRect(const Rect &r1, const Rect &r2)
+{
+ /* If either the first or the second is empty, return the other. */
+ if (IsEmptyRect(r1)) return r2;
+ if (IsEmptyRect(r2)) return r1;
+
+ Rect r;
+
+ r.top = std::min(r1.top, r2.top);
+ r.bottom = std::max(r1.bottom, r2.bottom);
+ r.left = std::min(r1.left, r2.left);
+ r.right = std::max(r1.right, r2.right);
+
+ return r;
+}
diff --git a/src/core/geometry_func.hpp b/src/core/geometry_func.hpp
index cd136488a..41df35945 100644
--- a/src/core/geometry_func.hpp
+++ b/src/core/geometry_func.hpp
@@ -14,4 +14,16 @@
Dimension maxdim(const Dimension &d1, const Dimension &d2);
+/**
+ * Check if a rectangle is empty.
+ * @param r Rectangle to check.
+ * @return True if and only if the rectangle doesn't define space.
+ */
+static inline bool IsEmptyRect(const Rect &r)
+{
+ return (r.left | r.top | r.right | r.bottom) == 0;
+}
+
+Rect BoundingRect(const Rect &r1, const Rect &r2);
+
#endif /* GEOMETRY_FUNC_HPP */