summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--graphics.c112
1 files changed, 73 insertions, 39 deletions
diff --git a/graphics.c b/graphics.c
index 50f3b68..ec86b42 100644
--- a/graphics.c
+++ b/graphics.c
@@ -26,6 +26,11 @@ enum {
OP_XOR,
};
+b8 rectangle_contains(f64 x0, f64 y0, f64 width, f64 height, f64 px, f64 py);
+b8 triangle_contains (f64 x0, f64 y0, f64 x1, f64 y1, f64 x2, f64 y2, f64 px, f64 py);
+b8 ellipse_contains (f64 x0, f64 y0, f64 width, f64 height, f64 px, f64 py);
+b8 line_contains (f64 x0, f64 y0, f64 x1, f64 y1, f64 width, f64 px, f64 py);
+
u32 u32_from_rgb (f32 red, f32 green, f32 blue);
void fill_rectangle (u32 op, u32 color, f64 x0, f64 y0, f64 width, f64 height);
void fill_triangle (u32 op, u32 color, f64 x0, f64 y0, f64 x1, f64 y1, f64 x2, f64 y2);
@@ -300,6 +305,70 @@ void put_pixel(i64 i, i64 j, u32 op, u32 color) {
platform.pixels[j * platform.frame_width + i] = color;
}
+b8 rectangle_contains(f64 x0, f64 y0, f64 width, f64 height, f64 px, f64 py) {
+ return px >= x0 && px < x0 + width && py >= y0 && py < y0 + height;
+}
+
+b8 triangle_contains(f64 x0, f64 y0, f64 x1, f64 y1, f64 x2, f64 y2, f64 px, f64 py) {
+ // Z-components of cross-products
+ //
+ f64 z0 = (x1 - x0) * (y2 - y0) - (x2 - x0) * (y1 - y0);
+ f64 z1 = (x2 - x1) * (y0 - y1) - (x0 - x1) * (y2 - y1);
+ f64 z2 = (x0 - x2) * (y1 - y2) - (x1 - x2) * (y0 - y2);
+
+ // Z-components of cross-products
+ //
+ f64 pz0 = (px - x0) * (y2 - y0) - (x2 - x0) * (py - y0);
+ f64 pz1 = (px - x1) * (y0 - y1) - (x0 - x1) * (py - y1);
+ f64 pz2 = (px - x2) * (y1 - y2) - (x1 - x2) * (py - y2);
+
+ // Check signs
+ //
+ return
+ same_sign(z0, pz0)
+ && same_sign(z1, pz1)
+ && same_sign(z2, pz2);
+}
+
+b8 ellipse_contains(f64 x0, f64 y0, f64 width, f64 height, f64 px, f64 py) {
+ f64 dw = width / 2;
+ f64 dh = height / 2;
+
+ if (dw < EPSILON || dh < EPSILON)
+ return 0;
+
+ f64 cx = x0 + dw;
+ f64 cy = y0 + dh;
+ f64 kx = 1. / dw;
+ f64 ky = 1. / dh;
+
+ f64 dx = (px - cx) * kx;
+ f64 dy = (py - cy) * ky;
+
+ return dx * dx + dy * dy - 1.0 < EPSILON;
+}
+
+b8 line_contains(f64 x0, f64 y0, f64 x1, f64 y1, f64 width, f64 px, f64 py) {
+ f64 dx = x1 - x0;
+ f64 dy = y1 - y0;
+
+ // Tangent
+ //
+ f64 tx = -dy;
+ f64 ty = dx;
+ f64 tl = sqrt(tx * tx + ty * ty);
+ if (tl >= EPSILON) {
+ tx /= tl;
+ ty /= tl;
+ }
+ tx *= width * .5;
+ ty *= width * .5;
+
+ return
+ triangle_contains(x0 - tx, y0 - ty, x0 + tx, y0 + ty, x1 + tx, y1 + ty, px, py)
+ || triangle_contains(x0 - tx, y0 - ty, x1 + tx, y1 + ty, x1 - tx, y1 - ty, px, py);
+}
+
void fill_rectangle(u32 op, u32 color, f64 x0, f64 y0, f64 width, f64 height) {
i64 i0 = (i64) floor(x0 + .5);
i64 j0 = (i64) floor(y0 + .5);
@@ -322,31 +391,10 @@ void fill_triangle(u32 op, u32 color, f64 x0, f64 y0, f64 x1, f64 y1, f64 x2, f6
i64 max_x = (i64) ceil (max3(x0, x1, x2));
i64 max_y = (i64) ceil (max3(y0, y1, y2));
- // Z-components of cross-products
- //
- f64 z0 = (x1 - x0) * (y2 - y0) - (x2 - x0) * (y1 - y0);
- f64 z1 = (x2 - x1) * (y0 - y1) - (x0 - x1) * (y2 - y1);
- f64 z2 = (x0 - x2) * (y1 - y2) - (x1 - x2) * (y0 - y2);
-
for (i64 j = min_y; j <= max_y; ++j)
- for (i64 i = min_x; i <= max_x; ++i) {
- f64 x = (f64) i;
- f64 y = (f64) j;
-
- // Z-components of cross-products
- //
- f64 pz0 = (x - x0) * (y2 - y0) - (x2 - x0) * (y - y0);
- f64 pz1 = (x - x1) * (y0 - y1) - (x0 - x1) * (y - y1);
- f64 pz2 = (x - x2) * (y1 - y2) - (x1 - x2) * (y - y2);
-
- // Check signs
- //
- if (!same_sign(z0, pz0)) continue;
- if (!same_sign(z1, pz1)) continue;
- if (!same_sign(z2, pz2)) continue;
-
- put_pixel(i, j, op, color);
- }
+ for (i64 i = min_x; i <= max_x; ++i)
+ if (triangle_contains(x0, y0, x1, y1, x2, y2, (f64) i, (f64) j))
+ put_pixel(i, j, op, color);
}
void fill_ellipse(u32 op, u32 color, f64 x0, f64 y0, f64 width, f64 height) {
@@ -358,27 +406,13 @@ void fill_ellipse(u32 op, u32 color, f64 x0, f64 y0, f64 width, f64 height) {
i64 i1 = (i64) floor(x0 + width + .5);
i64 j1 = (i64) floor(y0 + height + .5);
- f64 dw = width / 2;
- f64 dh = height / 2;
-
- if (dw < EPSILON || dh < EPSILON)
- return;
-
- f64 cx = x0 + dw;
- f64 cy = y0 + dh;
- f64 kx = 1. / dw;
- f64 ky = 1. / dh;
-
for (i64 j = j0; j < j1; ++j) {
if (j < 0 || j >= platform.frame_height)
continue;
- f64 dy = (((f64) j) - cy) * ky;
- f64 dydy = dy * dy;
for (i64 i = i0; i < i1; ++i) {
if (i < 0 || i >= platform.frame_width)
continue;
- f64 dx = (((f64) i) - cx) * kx;
- if (dx * dx + dydy <= 1.0)
+ if (ellipse_contains(x0, y0, width, height, (f64) i, (f64) j))
put_pixel(i, j, op, color);
}
}