| 1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
 | const cfg = @import("config.zig");
const std = @import("std");
const rl = @import("raylib");
const quad = @import("quad.zig");
pub const particle = struct {
    colorId: u32,
    pos: quad.Point,
    xvel: f32,
    yvel: f32,
};
/// Initialize a MultiArrayList of size amnt with particles created by createParticle
pub fn initParticles(allocator: std.mem.Allocator, amnt: u32) !std.ArrayList(particle) {
    var particles = std.ArrayList(particle).init(allocator);
    try particles.ensureTotalCapacity(cfg.particleMax);
    for (0..amnt) |_|
        try particles.append(createParticle());
    return particles;
}
/// Applies forces from the ruleset to each particle
pub fn updateVelocities(
    particles: std.ArrayList(particle),
    qtree: quad.Quad(particle, cfg.quadSplitLimit),
    threadidx: u64,
) !void {
    const rules = cfg.rules;
    var particlesInRange = std.ArrayList(particle).init(qtree.allocator);
    defer particlesInRange.deinit();
    var i = threadidx;
    while (i < particles.items.len) : (i += cfg.numThreads) {
        var p: *particle = &(particles.items[i]);
        defer particlesInRange.clearRetainingCapacity();
        const radius = cfg.radius[p.colorId];
        try qtree.radiusSearchWrapping(p.pos, @intCast(radius), &particlesInRange, rl.getScreenWidth(), rl.getScreenHeight());
        var forceX: f32 = 0.0;
        var forceY: f32 = 0.0;
        const floatRadius = @as(f32, @floatFromInt(radius));
        const floattMinDistance = @as(f32, @floatFromInt(cfg.minDistance));
        for (particlesInRange.items) |p2| {
            if (p.pos.x == p2.pos.x and p.pos.y == p2.pos.y) continue;
            const distance_x: f32 = @floatFromInt(p.pos.x - p2.pos.x);
            const distance_y: f32 = @floatFromInt(p.pos.y - p2.pos.y);
            var distance = @sqrt(distance_x * distance_x + distance_y * distance_y);
            if (distance == 0) distance = 0.01;
            const f = -force(distance, floatRadius, rules[p.colorId][p2.colorId]);
            forceX += (distance_x / distance) * f;
            forceY += (distance_y / distance) * f;
        }
        forceX = forceX * floattMinDistance / floatRadius;
        // forceX = std.math.clamp(forceX, -10.0, 10.0);
        forceY = forceY * floattMinDistance / floatRadius;
        // forceY = std.math.clamp(forceY, -10.0, 10.0);
        p.xvel *= cfg.friction ;
        p.xvel += forceX;
        p.yvel *= cfg.friction;
        p.yvel += forceY;
    }
}
/// Applies the particles velocity and updates position
pub fn updatePosition(particles: *std.ArrayList(particle)) void {
    for (particles.items) |*p| {
        const maxVel: f32 = 4096.0;
        const posYplusVel: f32 = @as(f32, @floatFromInt(p.pos.y)) + std.math.clamp(p.yvel, -maxVel, maxVel);
        const posXplusVel: f32 = @as(f32, @floatFromInt(p.pos.x)) + std.math.clamp(p.xvel, -maxVel, maxVel);
        p.pos.y = @mod(@as(i32, @intFromFloat(posYplusVel)),  rl.getScreenHeight());
        p.pos.x = @mod(@as(i32, @intFromFloat(posXplusVel)),  rl.getScreenWidth());
    }
}
/// Draw the particles onto the screen using raylib
pub fn draw(particles: std.ArrayList(particle)) void {
    for (particles.items) |p|
        rl.drawRectangle(p.pos.x, p.pos.y, 5, 5, cfg.colors[p.colorId]);
}
fn force(distance: f32, radius: f32, attraction: f32) f32 {
    const beta = @as(f32, @floatFromInt(cfg.minDistance)) / radius;
    const r: f32 = distance / radius;
    if (r < beta)
        return ((beta - r) / (beta - 1.0));
    if (beta <= r and r < 1)
        return attraction * (1 - @abs(2.0 * r - 1.0 - beta) / (1.0 - beta));
    return 0;
}
pub fn createParticle() particle {
    const seed = @as(u64, @truncate(@as(u128, @bitCast(std.time.nanoTimestamp()))));
    var prng = std.rand.DefaultPrng.init(seed);
    const x = prng.random().uintLessThan(u32, @intCast(rl.getScreenWidth()));
    const y = prng.random().uintLessThan(u32, @intCast(rl.getScreenHeight()));
    const color = prng.random().uintLessThan(u32, cfg.colorAmnt);
    return particle{
        .colorId = color,
        .pos = .{
            .x = @intCast(x),
            .y = @intCast(y),
        },
        .xvel = 0,
        .yvel = 0,
    };
}
//TODO: Create tests
test "Force values" {
    const expect = std.testing.expect;
    const radius = 50;
    cfg.minDistance = 20;
    const belowMin = force(5.0, radius, 0.5);
    const aboveMin = force(25.0, radius, 0.5);
    try expect(aboveMin > 0);
    try expect(belowMin < 0);
}
 |