stepInfection = function*(points) {
const originPoint = {x: 0, y: 0}
function angleBetween(a1, a2) {
const diff = Math.abs(a2 - a1) % (2 * Math.PI)
return diff < Math.PI ? diff : diff - Math.PI
}
function distance(x1, y1, x2, y2) {
return Math.sqrt(Math.pow(x2 - x1, 2) + Math.pow(y2 - y1, 2))
}
const maxDistance = 1.5 * (circleR + 2 * circleSpacing + 2 * variance)
const randomInfectionTime = d3.randomNormal(infectionTMu, infectionTSigma)
const randomInfectionCount = d3.randomNormal(infectionMMu, infectionMSigma)
const delaunay = d3.Delaunay.from(points, p => p.x, p => p.y)
const activeInfected = new Set(points.filter(p => p.infectionTime > 0))
function infect(p, by) {
let prevented = false
if (p.isCountermeasures && Math.random() < countermeasuresEffect) {
p.countermeasuresHits++
changedPoints.push(p)
prevented = true
}
if (by !== originPoint) {
const edgeKey = `${by.idx}->${p.idx}`
changedEdges.set(edgeKey, {
key: edgeKey,
x1: by.x,
y1: by.y,
x2: p.x,
y2: p.y,
halted: p.infected || prevented,
})
}
if (p.infected || prevented) {
return
}
p.infected = true
p.infectedBy = by || originPoint
p.infectionTime = Math.round(randomInfectionTime())
p.infectionCount = Math.round(randomInfectionCount())
by.infectionCount--
activeInfected.add(p)
changedPoints.push(p)
}
let tick = 0
let changedPoints = []
let changedEdges = new Map()
let stats = []
function stepData(pointsOverride) {
stats.push({
tick: tick++,
infectedCount: activeInfected.size,
})
return {
changedPoints: pointsOverride || changedPoints,
changedEdges: [...changedEdges.values()],
stats,
}
}
yield stepData(points)
// Infect initial ring.
points.filter(p => p.ring === ringStart).forEach(p => infect(p, originPoint))
yield stepData(changedPoints)
while (activeInfected.size) {
changedPoints = []
changedEdges = new Map()
for (const p of [...activeInfected]) {
changedPoints.push(p)
p.infectionTime--
if (p.infectionTime <= 0) {
activeInfected.delete(p)
continue
}
if (p.ring === ringCount + ringStart - 1) {
continue
}
const neighbors = _.shuffle([...delaunay.neighbors(p.idx)])
for (const neighborIdx of neighbors) {
if (p.infectionCount <= 0) {
break
}
const neighbor = points[neighborIdx]
if (neighbor === p.infectedBy) {
// No tag-backs.
continue
}
if (distance(p.x, p.y, neighbor.x, neighbor.y) > maxDistance) {
continue
}
const byAngle = Math.atan2(p.y - p.infectedBy.y, p.x - p.infectedBy.x)
const toAngle = Math.atan2(neighbor.y - p.y, neighbor.x - p.x)
const similarity = 1 - (angleBetween(byAngle, toAngle) / Math.PI)
if (similarity < angleThreshold[1] && Math.random() > (similarity - angleThreshold[0]) * (angleThreshold[1] - angleThreshold[0])) {
continue
}
if (Math.random() < infectionP) {
infect(neighbor, p)
}
}
}
yield stepData()
}
}