summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorAndreas Kling <kling@serenityos.org>2020-04-12 19:19:35 +0200
committerAndreas Kling <kling@serenityos.org>2020-04-12 19:23:39 +0200
commit6f2c63000df2c5005f9c99d57b65542460515175 (patch)
tree3bea1aad2bfd335d557a6051585b929a5b5bbd52
parent3bbc2c73007b92c982304c5d70d20ef41c083aac (diff)
downloadserenity-6f2c63000df2c5005f9c99d57b65542460515175.zip
LibGfx: Add a basic AffineTransform class
We can now perform some basic 2D transforms through an affine matrix. This patch adds translate() and scale() :^)
-rw-r--r--Libraries/LibGfx/AffineTransform.cpp148
-rw-r--r--Libraries/LibGfx/AffineTransform.h74
-rw-r--r--Libraries/LibGfx/Makefile1
3 files changed, 223 insertions, 0 deletions
diff --git a/Libraries/LibGfx/AffineTransform.cpp b/Libraries/LibGfx/AffineTransform.cpp
new file mode 100644
index 0000000000..976686d03b
--- /dev/null
+++ b/Libraries/LibGfx/AffineTransform.cpp
@@ -0,0 +1,148 @@
+/*
+ * Copyright (c) 2020, Andreas Kling <kling@serenityos.org>
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright notice, this
+ * list of conditions and the following disclaimer.
+ *
+ * 2. Redistributions in binary form must reproduce the above copyright notice,
+ * this list of conditions and the following disclaimer in the documentation
+ * and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
+ * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
+ * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
+ * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#include <AK/LogStream.h>
+#include <AK/Optional.h>
+#include <LibGfx/AffineTransform.h>
+#include <LibGfx/FloatRect.h>
+#include <LibGfx/Rect.h>
+
+namespace Gfx {
+
+bool AffineTransform::is_identity() const
+{
+ return m_values[0] == 1 && m_values[1] == 0 && m_values[2] == 0 && m_values[3] == 1 && m_values[4] == 0 && m_values[5] == 0;
+}
+
+static float hypotenuse(float x, float y)
+{
+ // FIXME: This won't handle overflow :(
+ return sqrt(x * x + y * y);
+}
+
+float AffineTransform::x_scale() const
+{
+ return hypotenuse(m_values[0], m_values[1]);
+}
+
+float AffineTransform::y_scale() const
+{
+ return hypotenuse(m_values[2], m_values[3]);
+}
+
+AffineTransform& AffineTransform::scale(float sx, float sy)
+{
+ m_values[0] *= sx;
+ m_values[1] *= sx;
+ m_values[2] *= sy;
+ m_values[3] *= sy;
+ return *this;
+}
+
+AffineTransform& AffineTransform::translate(float tx, float ty)
+{
+ m_values[4] += tx * m_values[0] + ty * m_values[2];
+ m_values[5] += tx * m_values[1] + ty * m_values[3];
+ return *this;
+}
+
+void AffineTransform::map(float unmapped_x, float unmapped_y, float& mapped_x, float& mapped_y) const
+{
+ mapped_x = (m_values[0] * unmapped_x + m_values[2] * unmapped_y + m_values[4]);
+ mapped_y = (m_values[1] * unmapped_x + m_values[3] * unmapped_y + m_values[5]);
+}
+
+Point AffineTransform::map(const Point& point) const
+{
+ float mapped_x;
+ float mapped_y;
+ map(point.x(), point.y(), mapped_x, mapped_y);
+ return Point(roundf(mapped_x), roundf(mapped_y));
+}
+
+FloatPoint AffineTransform::map(const FloatPoint& point) const
+{
+ float mapped_x;
+ float mapped_y;
+ map(point.x(), point.y(), mapped_x, mapped_y);
+ return FloatPoint(mapped_x, mapped_y);
+}
+
+Size AffineTransform::map(const Size& size) const
+{
+ return Size(roundf(size.width() * x_scale()), roundf(y_scale()));
+}
+
+FloatSize AffineTransform::map(const FloatSize& size) const
+{
+ return { size.width() * x_scale(), size.height() * y_scale() };
+}
+
+Rect AffineTransform::map(const Rect& rect) const
+{
+ return enclosing_int_rect(map(FloatRect(rect)));
+}
+
+template<typename T>
+static T smallest_of(T p1, T p2, T p3, T p4)
+{
+ return min(min(p1, p2), min(p3, p4));
+}
+
+template<typename T>
+static T largest_of(T p1, T p2, T p3, T p4)
+{
+ return max(max(p1, p2), max(p3, p4));
+}
+
+FloatRect AffineTransform::map(const FloatRect& rect) const
+{
+ FloatPoint p1 = map(rect.top_left());
+ FloatPoint p2 = map(rect.top_right().translated(1, 0));
+ FloatPoint p3 = map(rect.bottom_right().translated(1, 1));
+ FloatPoint p4 = map(rect.bottom_left().translated(0, 1));
+ float left = smallest_of(p1.x(), p2.x(), p3.x(), p4.x());
+ float top = smallest_of(p1.y(), p2.y(), p3.y(), p4.y());
+ float right = largest_of(p1.x(), p2.x(), p3.x(), p4.x());
+ float bottom = largest_of(p1.y(), p2.y(), p3.y(), p4.y());
+ return { left, top, right - left, bottom - top };
+}
+
+const LogStream& operator<<(const LogStream& stream, const AffineTransform& value)
+{
+ if (value.is_identity())
+ return stream << "{ Identity }";
+
+ return stream << "{ "
+ << value.a() << ", "
+ << value.b() << ", "
+ << value.c() << ", "
+ << value.d() << ", "
+ << value.e() << ", "
+ << value.f() << " }";
+}
+
+}
diff --git a/Libraries/LibGfx/AffineTransform.h b/Libraries/LibGfx/AffineTransform.h
new file mode 100644
index 0000000000..9cc2b080d7
--- /dev/null
+++ b/Libraries/LibGfx/AffineTransform.h
@@ -0,0 +1,74 @@
+/*
+ * Copyright (c) 2020, Andreas Kling <kling@serenityos.org>
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright notice, this
+ * list of conditions and the following disclaimer.
+ *
+ * 2. Redistributions in binary form must reproduce the above copyright notice,
+ * this list of conditions and the following disclaimer in the documentation
+ * and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
+ * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
+ * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
+ * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#pragma once
+
+#include <AK/Forward.h>
+#include <AK/LogStream.h>
+#include <LibGfx/Forward.h>
+
+namespace Gfx {
+
+class AffineTransform {
+public:
+ AffineTransform()
+ : m_values { 1, 0, 0, 1, 0, 0 }
+ {
+ }
+
+ bool is_identity() const;
+
+ void map(float unmapped_x, float unmapped_y, float& mapped_x, float& mapped_y) const;
+
+ Point map(const Point&) const;
+ FloatPoint map(const FloatPoint&) const;
+
+ Size map(const Size&) const;
+ FloatSize map(const FloatSize&) const;
+
+ Rect map(const Rect&) const;
+ FloatRect map(const FloatRect&) const;
+
+ float a() const { return m_values[0]; }
+ float b() const { return m_values[1]; }
+ float c() const { return m_values[2]; }
+ float d() const { return m_values[3]; }
+ float e() const { return m_values[4]; }
+ float f() const { return m_values[5]; }
+
+ float x_scale() const;
+ float y_scale() const;
+
+ AffineTransform& scale(float sx, float sy);
+ AffineTransform& translate(float tx, float ty);
+
+private:
+ float m_values[6] { 0 };
+};
+
+const LogStream& operator<<(const LogStream&, const AffineTransform&);
+
+}
diff --git a/Libraries/LibGfx/Makefile b/Libraries/LibGfx/Makefile
index 4c538660c7..2645693cee 100644
--- a/Libraries/LibGfx/Makefile
+++ b/Libraries/LibGfx/Makefile
@@ -1,4 +1,5 @@
OBJS = \
+ AffineTransform.o \
Bitmap.o \
CharacterBitmap.o \
Color.o \