diff options
author | Fbenas <philbeansburton@gmail.com> | 2020-10-13 21:58:58 +0100 |
---|---|---|
committer | Fbenas <philbeansburton@gmail.com> | 2020-10-13 21:58:58 +0100 |
commit | a44e609b27642afc77495293d975f2ea0bbf6f3e (patch) | |
tree | fe8f4d8ed8ca9f2d75e8568a0a5395a12c1eb0bc | |
parent | 56f42daf6aadb940c55d46b612c46f9afc3211fa (diff) |
Add maths function to detect two intersection circles
-rw-r--r-- | public/js/app.js | 138 | ||||
-rw-r--r-- | resources/js/boid.js | 138 |
2 files changed, 216 insertions, 60 deletions
diff --git a/public/js/app.js b/public/js/app.js index 093d6df..8ba4cf8 100644 --- a/public/js/app.js +++ b/public/js/app.js @@ -36,11 +36,23 @@ class Boid { let rayInteval = 2; let noOfSteps = this.fieldOfView / rayInteval; + let is_clockwise = Math.random() >= 0.5; + for (let i = 0; i < noOfSteps / 2; i++) { - if (i != 0) { + if (i == 0) { + rays.push(rayInteval * i); + continue; + } + + if (is_clockwise) { rays.push(rayInteval * -i); } + rays.push(rayInteval * i); + + if (!is_clockwise) { + rays.push(rayInteval * -i); + } } return rays; @@ -57,10 +69,12 @@ class Boid { let rayAngle = tweakAngle + this.direction + rays[i]; - if (i == 0 && this.detectBoid(rayAngle, boids)) { + // Check against all other boids + if (i == 0 && this.detectBoids(rayAngle, boids)) { continue; } + // Check against scene boundaries if (this.detectBox(this.scene_width, this.scene_height, rayAngle)) { continue; } @@ -71,34 +85,9 @@ class Boid { console.log('cannot find suitable ray'); } - detectBoid(direction, boids) { + detectBoids(direction, boids) { for (let i = 0; i < boids.length; i++) { - // rule out ourselves - if (this.id == boids[i].id) { - continue; - } - - let thisFututrePosition = this.findPoint(this.x, this.y, this.boidBuffer, direction); - let thisPath = { - x1: this.x, - y1: this.y, - x2: thisFututrePosition.x, - y2: thisFututrePosition.y - }; - - let boidFuturePosition = this.findPoint(boids[i].x, boids[i].y, boids[i].boidBuffer, boids[i].direction); - let boidPath = { - x1: boids[i].x, - y1: boids[i].y, - x2: boidFuturePosition.x, - y2: boidFuturePosition.y - }; - - let thisIntersectsBoid = this.pathsIntersect( - thisPath.x1, thisPath.y1, thisPath.x2, thisPath.y2, boidPath.x1, boidPath.y1, boidPath.x2, boidPath.y2 - ); - - if (thisIntersectsBoid) { + if (this.detectBoid(direction, boids[i])) { return true; } } @@ -106,6 +95,95 @@ class Boid { return false; } + detectBoid(direction, boid) { + // rule out ourselves + if (this.id == boid.id) { + return false; + } + + // let cross = this.doPathsCross(direction, boid); + // + // if (cross) { + // console.log('paths cross detected'); + // } + + let intersect = this.boidsIntersect(boid); + + if (intersect) { + console.log('boids intersect detected'); + } + } + + boidsIntersect(boid) { + this.twoCirclesIntersect(this.radius, boid.radius, this.x, boid.x, this.y, boid.y); + } + + twoCirclesIntersect(r1, r2, x1, x2, y1, y2) { + // from https://planetcalc.com/8098/ (bottom of page) + // https://mathworld.wolfram.com/Circle-CircleIntersection.html + // d is the abs distance between the two circles center + let d = this.distanceBetweenPoints(x1, x2, y1, y2); + + // a is distance to our x3 point ( I think ) + let a = (Math.pow(r1, 2) - Math.pow(r2, 2) + Math.pow(d, 2)) / (2 * d); + + // h is some other segment of some other circle ( I dont really know ) + let h = Math.sqrt(Math.pow(r1, 2) - Math.pow(a, 2)); + + // x3 and y3 represent the point between the two circle midpoints, perpindicular to which will be our intersects + // ( maybe ) + let x3 = x1 + ((a / d) * (x2 - x1)); + let y3 = y1 + ((a / d) * (y2 - y1)); + + // x4 and y4 are one of the intersect points + let x4 = x3 + ((h / d) * (y2 - y1)); + let y4 = y3 - ((h / d) * (x2 - x1)); + + if (!x4 || !y4) { + return false; + } + console.log('intersecting circles'); + return { + x: x4, + y: y4 + }; + } + + distanceBetweenPoints(x1, x2, y1, y2) { + let dx = Math.abs(x1 - x2); + let dy = Math.abs(y1 - y2); + + return Math.sqrt(Math.pow(dx, 2) + Math.pow(dy, 2)); + } + + doPathsCross(direction, boid) { + let thisFututrePosition = this.findPoint(this.x, this.y, this.boidBuffer, direction); + let thisPath = { + x1: this.x, + y1: this.y, + x2: thisFututrePosition.x, + y2: thisFututrePosition.y + }; + + let boidFuturePosition = this.findPoint(boid.x, boid.y, boid.boidBuffer, boid.direction); + let boidPath = { + x1: boid.x, + y1: boid.y, + x2: boidFuturePosition.x, + y2: boidFuturePosition.y + }; + + let thisIntersectsBoid = this.pathsIntersect( + thisPath.x1, thisPath.y1, thisPath.x2, thisPath.y2, boidPath.x1, boidPath.y1, boidPath.x2, boidPath.y2 + ); + + if (thisIntersectsBoid) { + return true; + } + + return false; + } + pathsIntersect(x1, y1, x2, y2, x3, y3, x4, y4) { // Check if none of the lines are of length 0 if ((x1 === x2 && y1 === y2) || (x3 === x4 && y3 === y4)) { @@ -165,7 +243,7 @@ class Boid { draw(context) { this.drawBoid(context); - this.drawRay(context, this.x, this.y, this.rayLength, this.direction); + // this.drawRay(context, this.x, this.y, this.rayLength, this.direction); } drawBoid(context) { diff --git a/resources/js/boid.js b/resources/js/boid.js index 2ae692d..85793ff 100644 --- a/resources/js/boid.js +++ b/resources/js/boid.js @@ -36,11 +36,23 @@ class Boid { let rayInteval = 2; let noOfSteps = this.fieldOfView / rayInteval; + let is_clockwise = Math.random() >= 0.5; + for (let i = 0; i < noOfSteps / 2; i++) { - if (i != 0) { + if (i == 0) { + rays.push(rayInteval * i); + continue; + } + + if (is_clockwise) { rays.push(rayInteval * -i); } + rays.push(rayInteval * i); + + if (!is_clockwise) { + rays.push(rayInteval * -i); + } } return rays; @@ -57,10 +69,12 @@ class Boid { let rayAngle = tweakAngle + this.direction + rays[i]; - if (i == 0 && this.detectBoid(rayAngle, boids)) { + // Check against all other boids + if (i == 0 && this.detectBoids(rayAngle, boids)) { continue; } + // Check against scene boundaries if (this.detectBox(this.scene_width, this.scene_height, rayAngle)) { continue; } @@ -71,34 +85,9 @@ class Boid { console.log('cannot find suitable ray'); } - detectBoid(direction, boids) { + detectBoids(direction, boids) { for (let i = 0; i < boids.length; i++) { - // rule out ourselves - if (this.id == boids[i].id) { - continue; - } - - let thisFututrePosition = this.findPoint(this.x, this.y, this.boidBuffer, direction); - let thisPath = { - x1: this.x, - y1: this.y, - x2: thisFututrePosition.x, - y2: thisFututrePosition.y - }; - - let boidFuturePosition = this.findPoint(boids[i].x, boids[i].y, boids[i].boidBuffer, boids[i].direction); - let boidPath = { - x1: boids[i].x, - y1: boids[i].y, - x2: boidFuturePosition.x, - y2: boidFuturePosition.y - }; - - let thisIntersectsBoid = this.pathsIntersect( - thisPath.x1, thisPath.y1, thisPath.x2, thisPath.y2, boidPath.x1, boidPath.y1, boidPath.x2, boidPath.y2 - ); - - if (thisIntersectsBoid) { + if (this.detectBoid(direction, boids[i])) { return true; } } @@ -106,6 +95,95 @@ class Boid { return false; } + detectBoid(direction, boid) { + // rule out ourselves + if (this.id == boid.id) { + return false; + } + + // let cross = this.doPathsCross(direction, boid); + // + // if (cross) { + // console.log('paths cross detected'); + // } + + let intersect = this.boidsIntersect(boid); + + if (intersect) { + console.log('boids intersect detected'); + } + } + + boidsIntersect(boid) { + this.twoCirclesIntersect(this.radius, boid.radius, this.x, boid.x, this.y, boid.y); + } + + twoCirclesIntersect(r1, r2, x1, x2, y1, y2) { + // from https://planetcalc.com/8098/ (bottom of page) + // https://mathworld.wolfram.com/Circle-CircleIntersection.html + // d is the abs distance between the two circles center + let d = this.distanceBetweenPoints(x1, x2, y1, y2); + + // a is distance to our x3 point ( I think ) + let a = (Math.pow(r1, 2) - Math.pow(r2, 2) + Math.pow(d, 2)) / (2 * d); + + // h is some other segment of some other circle ( I dont really know ) + let h = Math.sqrt(Math.pow(r1, 2) - Math.pow(a, 2)); + + // x3 and y3 represent the point between the two circle midpoints, perpindicular to which will be our intersects + // ( maybe ) + let x3 = x1 + ((a / d) * (x2 - x1)); + let y3 = y1 + ((a / d) * (y2 - y1)); + + // x4 and y4 are one of the intersect points + let x4 = x3 + ((h / d) * (y2 - y1)); + let y4 = y3 - ((h / d) * (x2 - x1)); + + if (!x4 || !y4) { + return false; + } + console.log('intersecting circles'); + return { + x: x4, + y: y4 + }; + } + + distanceBetweenPoints(x1, x2, y1, y2) { + let dx = Math.abs(x1 - x2); + let dy = Math.abs(y1 - y2); + + return Math.sqrt(Math.pow(dx, 2) + Math.pow(dy, 2)); + } + + doPathsCross(direction, boid) { + let thisFututrePosition = this.findPoint(this.x, this.y, this.boidBuffer, direction); + let thisPath = { + x1: this.x, + y1: this.y, + x2: thisFututrePosition.x, + y2: thisFututrePosition.y + }; + + let boidFuturePosition = this.findPoint(boid.x, boid.y, boid.boidBuffer, boid.direction); + let boidPath = { + x1: boid.x, + y1: boid.y, + x2: boidFuturePosition.x, + y2: boidFuturePosition.y + }; + + let thisIntersectsBoid = this.pathsIntersect( + thisPath.x1, thisPath.y1, thisPath.x2, thisPath.y2, boidPath.x1, boidPath.y1, boidPath.x2, boidPath.y2 + ); + + if (thisIntersectsBoid) { + return true; + } + + return false; + } + pathsIntersect(x1, y1, x2, y2, x3, y3, x4, y4) { // Check if none of the lines are of length 0 if ((x1 === x2 && y1 === y2) || (x3 === x4 && y3 === y4)) { @@ -165,7 +243,7 @@ class Boid { draw(context) { this.drawBoid(context); - this.drawRay(context, this.x, this.y, this.rayLength, this.direction); + // this.drawRay(context, this.x, this.y, this.rayLength, this.direction); } drawBoid(context) { |