Hello everybody!
Recently I have got invested in boids. I have made 90% of the system, but I feel that it is messy/unoptimized and I have yet to figure out a way to apply the third principle, separation. The code is well documented so you should hopefully be able to figure it out.
TL; DR. Need help finishing my code and cleaning up bad practices!
Github: https://github.com/EternalEther/BoidSimulation/tree/main
Wikipedia: Boids - Wikipedia
local boidClass = {}
boidClass.__index = boidClass
local config = require(script.Config)
local collectionService = game:GetService("CollectionService")
function boidClass:_construct()
-- create a boid with a physical obj
local boid = setmetatable({}, boidClass)
boid.object = boidClass._setup()
return boid
end
function boidClass._setup()
local new_boid = script.Boid:Clone() -- grab boid object and clone it
new_boid.Parent = game.Workspace
new_boid.Position = Vector3.new(math.random(-10, 10), 10, math.random(-10, 10))
new_boid.CFrame = CFrame.lookAt(new_boid.Position, Vector3.new(math.random(-5, 5), 10, math.random(-5, 5)))
collectionService:AddTag(new_boid, "Boid") -- tag object as Boid
return new_boid
end
function boidClass:_localBoids()
-- return boids within a 6-stud radius
local function filterForBoids(filter_table)
local boidTable = {}
for _, object in ipairs(filter_table) do
if collectionService:HasTag(object, "Boid") then -- check if object is a boid
table.insert(boidTable, object)
end
end
return boidTable -- return the table, now boids only
end
local fieldOfVision = Vector3.new(12, 12, 12) -- radius is half of diameter, 6-studs in each direction
--print(self.object.Position)
--print(self.object.CFrame)
local touching = game.Workspace:GetPartBoundsInBox(self.object.CFrame, fieldOfVision)
return filterForBoids(touching) -- return table of boids, or nil
end
function boidClass:_centerOfFlock(boid_table)
local avgPosition = Vector3.new(0, 0, 0)
for _, boid in ipairs(boid_table) do
avgPosition += boid.Position
end
return avgPosition / #boid_table
end
function boidClass:_alignedCenter(center_position, boid_table)
local avgAlignment = Vector3.new(0, 0, 0)
for _, boid in ipairs(boid_table) do
avgAlignment += boid.CFrame.lookVector * config.boid.speed
end
return ((avgAlignment / #boid_table) + (center_position)) / 2
end
function boidClass:_update()
-- update boid position/cframe
local localBoids = self:_localBoids() -- boids within the local space
local centerOfFlock = self:_centerOfFlock(localBoids) -- average position of local boids
local alignedCenter = self:_alignedCenter(centerOfFlock, localBoids) -- average between flock direction and center position
self.object.CFrame = CFrame.lookAt(self.object.Position, centerOfFlock) -- point the boid towards midway between next projected position and flock center
self.object.Position += self.object.CFrame.lookVector * config.boid.speed -- move boid forward by config speed
self.object.Position = Vector3.new(math.clamp(self.object.Position.X, -25, 25), math.clamp(self.object.Position.Y, 25, 25), math.clamp(self.object.Position.Z, -50, 50)) -- clamp boid position within bounds
-- line 64 = cohesion + alignment
end
return boidClass
Feel free to submit pull requests and I will review them. Any help with cleaning up my code or adding the third principle for boids! Thanks for reading.