aboutsummaryrefslogtreecommitdiff
path: root/src/quad.zig
diff options
context:
space:
mode:
authorNic Gaffney <gaffney_nic@protonmail.com>2024-10-18 18:58:08 -0500
committerNic Gaffney <gaffney_nic@protonmail.com>2024-10-18 18:58:08 -0500
commitae555d16a0a355563c7da3d95c8ebbd192f52cf2 (patch)
tree444df3b075e6422588b48cbb567c59473ea48cf3 /src/quad.zig
parent5a0234c0eff069e13fdef204d810d994ab7858f9 (diff)
downloadparticle-sim-ae555d16a0a355563c7da3d95c8ebbd192f52cf2.tar.gz
Continue work on quadtree
Diffstat (limited to 'src/quad.zig')
-rw-r--r--src/quad.zig146
1 files changed, 146 insertions, 0 deletions
diff --git a/src/quad.zig b/src/quad.zig
new file mode 100644
index 0000000..92e345c
--- /dev/null
+++ b/src/quad.zig
@@ -0,0 +1,146 @@
+const std = @import("std");
+
+const Point = struct {
+ x: i32,
+ y: i32,
+};
+
+pub fn Node(T: type) type {
+ return struct {
+ pos: Point,
+ data: T,
+ };
+}
+
+pub fn Quad(T: type) type {
+ return struct {
+ allocator: std.mem.Allocator,
+ node: ?*Node(T),
+ topLeft: Point,
+ bottomRight: Point,
+ children: [4]?*Quad(T),
+
+ pub fn init(allocator: std.mem.Allocator, tl: Point, br: Point) Quad(T) {
+ return Quad(T){
+ .allocator = allocator,
+ .node = null,
+ .topLeft = tl,
+ .bottomRight = br,
+ .children = [4]?*Quad(T){ null, null, null, null },
+ };
+ }
+
+ inline fn inBoundry(self: Quad(T), pos: Point) bool {
+ return pos.x >= self.topLeft.x and pos.x <= self.bottomRight.x and pos.y >= self.topLeft.y and pos.y <= self.bottomRight.y;
+ }
+
+ pub fn search(self: Quad(T), p: Point) ?*Node(T) {
+ if (!self.inBoundry(p)) return null;
+ if (self.node) |n| return n;
+ if (@divTrunc((self.topLeft.x + self.bottomRight.x), 2) >= p.x) {
+ if (@divTrunc((self.topLeft.y + self.bottomRight.y), 2) >= p.y) {
+ if (self.children[0] == null)
+ return null;
+ return self.children[0].?.search(p);
+ } else {
+ if (self.children[2] == null)
+ return null;
+ return self.children[2].?.search(p);
+ }
+ } else {
+ if (@divTrunc((self.topLeft.y + self.bottomRight.y), 2) >= p.y) {
+ if (self.children[1] == null)
+ return null;
+ return self.children[1].?.search(p);
+ } else {
+ if (self.children[3] == null)
+ return null;
+ return self.children[3].?.search(p);
+ }
+ }
+ }
+
+ pub fn insert(self: *Quad(T), data: T, pos: Point) !void {
+ const node: ?*Node(T) = try self.allocator.create(Node(T));
+ node.* = Node(T){ .data = data, .pos = pos };
+ const nNode = node.?;
+ if (!self.inBoundry(nNode.pos)) return;
+ if (@abs(self.topLeft.x - self.bottomRight.x) <= 1 and @abs(self.topLeft.y - self.bottomRight.y) <= 1) {
+ self.node = if (self.node == null) node else self.node;
+ return;
+ }
+ if (@divTrunc((self.topLeft.x + self.bottomRight.x), 2) >= nNode.pos.x) {
+ if (@divTrunc((self.topLeft.y + self.bottomRight.y), 2) >= nNode.pos.y) {
+ if (self.children[0] == null) {
+ self.children[0] = try self.allocator.create(Quad(T));
+ self.children[0].?.* = Quad(T).init(
+ self.allocator,
+ self.topLeft,
+ .{
+ .x = @divTrunc((self.topLeft.x + self.bottomRight.x), 2),
+ .y = @divTrunc((self.topLeft.y + self.bottomRight.y), 2),
+ },
+ );
+ }
+ try self.children[0].?.insert(node);
+ } else {
+ if (self.children[2] == null) {
+ self.children[2] = try self.allocator.create(Quad(T));
+ self.children[2].?.* = Quad(T).init(
+ self.allocator,
+ .{
+ .x = self.topLeft.x,
+ .y = @divTrunc(self.topLeft.y + self.bottomRight.y, 2),
+ },
+ .{
+ .x = @divTrunc((self.topLeft.x + self.bottomRight.x), 2),
+ .y = self.bottomRight.y,
+ },
+ );
+ }
+ try self.children[2].?.insert(node);
+ }
+ } else {
+ if (@divTrunc((self.topLeft.y + self.bottomRight.y), 2) >= nNode.pos.y) {
+ if (self.children[1] == null) {
+ self.children[1] = try self.allocator.create(Quad(T));
+ self.children[1].?.* = Quad(T).init(
+ self.allocator,
+ .{
+ .x = @divTrunc(self.topLeft.x + self.bottomRight.x, 2),
+ .y = self.topLeft.y,
+ },
+ .{
+ .x = self.bottomRight.x,
+ .y = @divTrunc((self.topLeft.y + self.bottomRight.y), 2),
+ },
+ );
+ }
+ try self.children[1].?.insert(node);
+ } else {
+ if (self.children[3] == null) {
+ self.children[3] = try self.allocator.create(Quad(T));
+ self.children[3].?.* = Quad(T).init(
+ self.allocator,
+ .{
+ .x = @divTrunc((self.topLeft.x + self.bottomRight.x), 2),
+ .y = @divTrunc((self.topLeft.y + self.bottomRight.y), 2),
+ },
+ self.bottomRight,
+ );
+ }
+ try self.children[3].?.insert(node);
+ }
+ }
+ }
+
+ pub fn deinit(self: *Quad(T)) void {
+ if (self.node) |n| self.allocator.destroy(n);
+ for (self.children) |child|
+ if (child) |c| {
+ c.deinit();
+ self.allocator.destroy(c);
+ };
+ }
+ };
+}