#javascript #collision-detection #p5.js
#javascript #обнаружение столкновений #p5.js
Вопрос:
У меня есть игра asteroid, в которой игрок управляет объектом космического корабля и стреляет пулями по астероидам, чтобы уничтожить их. Маркеры будут удалены из массива маркеров, если они покинут экран. Иногда, во время тестирования программы, я получаю сообщение об ошибке, в котором говорится:
Uncaught TypeError: Cannot read property 'x' of undefined
at isInside (sketch.js:90)
at checkCollisions (sketch.js:80)
at draw (sketch.js:32)
at b.a.default.redraw (p5.min.js:3)
at _draw (p5.min.js:3)
Кажется, это часто случается, если я просто произвольно выпускаю кучу пуль. Проблема, по-видимому, связана с обнаружением столкновения между астероидом и пулей. Я предполагаю, что некоторые пули попали в астероид за кадром, но пуля была удалена, поэтому она не может правильно рассчитать столкновение? Я не слишком уверен. Вот код для игры в asteroid.
BulletSystem
class BulletSystem {
constructor(){
this.bullets = [];
this.velocity = new createVector(0, -5);
this.diam = 10;
}
run(){
this.move();
this.draw();
this.edges();
}
fire(x, y){
this.bullets.push(createVector(x,y));
print(this.bullets.length);
}
//draws all bullets
draw(){
for (var i=0; i<this.bullets.length; i ){
fill("#00e5ff");
ellipse(this.bullets[i].x, this.bullets[i].y, this.diam, this.diam);
}
}
//updates the location of all bullets
move(){
for (var i=0; i<this.bullets.length; i ){
this.bullets[i].y = this.velocity.y;
}
}
//check if bullets leave the screen and remove them from the array
edges(){
for (var i=this.bullets.length-1; i>=0; i--){
if(this.bullets[i].y < 0){
this.bullets.splice(i, 1);
}
}
}
}
Астероидная система
class AsteroidSystem {
//creates arrays to store each asteroid's data
constructor(){
this.locations = [];
this.velocities = [];
this.accelerations = [];
this.diams = [];
this.score = 0;
this.difficulty = 0;
}
run(){
this.spawn();
this.move();
this.draw();
}
// spawns asteroid at random intervals
spawn(){
if (random(1)<0.01 this.difficulty){ //spawn frequency increases with time
this.accelerations.push(new createVector(0,random(0.1,1)));
this.velocities.push(new createVector(0, 0));
this.locations.push(new createVector(random(width), 0));
this.diams.push(random(30,50));
this.difficulty =0.0001;
}
}
//moves all asteroids
move(){
for (var i=0; i<this.locations.length; i ){
this.velocities[i].add(this.accelerations[i]);
this.locations[i].add(this.velocities[i]);
this.accelerations[i].mult(0);
}
}
applyForce(f){
for (var i=0; i<this.locations.length; i ){
this.accelerations[i].add(f);
}
}
//draws all asteroids
draw(){
noStroke();
fill(100);
for (var i=0; i<this.locations.length; i ){
//this.polygon(this.locations[i].x, this.locations[i].y, this.diams[i]*0.5, 8); //create octagon
image(asteroidimg, this.locations[i].x, this.locations[i].y, this.diams[i], this.diams[i]); //create images of asteroid
}
}
//function that calculates effect of gravity on each asteroid and accelerates it
calcGravity(centerOfMass){
for (var i=0; i<this.locations.length; i ){
var gravity = p5.Vector.sub(centerOfMass, this.locations[i]);
gravity.normalize();
gravity.mult(.001);
this.applyForce(gravity);
}
}
//destroys all data associated with each asteroid
destroy(index){
this.locations.splice(index,1);
this.velocities.splice(index,1);
this.accelerations.splice(index,1);
this.diams.splice(index,1);
this.score ;
}
//function to create polygons
polygon(x, y, radius, npoints) {
let angle = TWO_PI / npoints;
beginShape();
for (let a = 0; a < TWO_PI; a = angle) {
let sx = x cos(a) * radius;
let sy = y sin(a) * radius;
vertex(sx, sy);
}
endShape(CLOSE);
}
}
Основной файл
var spaceship;
var asteroids;
var atmosphereLoc;
var atmosphereSize;
var earthLoc;
var earthSize;
var starLocs = [];
//////////////////////////////////////////////////
function setup() {
createCanvas(1200,800);
spaceship = new Spaceship();
asteroids = new AsteroidSystem();
//location and size of earth and its atmosphere
atmosphereLoc = new createVector(width/2, height*2.9);
atmosphereSize = new createVector(width*3, width*3);
earthLoc = new createVector(width/2, height*3.1);
earthSize = new createVector(width*3, width*3);
}
//////////////////////////////////////////////////
function draw() {
background(0);
sky();
spaceship.run();
asteroids.run();
drawEarth();
checkCollisions(spaceship, asteroids); // function that checks collision between various elements
score(); //function that keeps score of how many asteroids have been hit
}
//////////////////////////////////////////////////
//draws earth and atmosphere
function drawEarth(){
noStroke();
//draw atmosphere
fill(150,150,150,50);
ellipse(atmosphereLoc.x, atmosphereLoc.y, atmosphereSize.x, atmosphereSize.y);
//draw earth
fill(0,0,200);
ellipse(earthLoc.x, earthLoc.y, earthSize.x, earthSize.y);
}
//////////////////////////////////////////////////
//checks collisions between all types of bodies
function checkCollisions(spaceship, asteroids){
//spaceship-2-asteroid collisions
for (var i=0; i<asteroids.locations.length; i ){
if (isInside(spaceship.location, spaceship.size, asteroids.locations[i], asteroids.diams[i]*2)==true){
gameOver();
}
}
//asteroid-2-earth collisions
for (var i=0; i<asteroids.locations.length; i ){
if (isInside(earthLoc, earthSize.x, asteroids.locations[i], asteroids.diams[i]*2)==true){
gameOver();
}
}
//spaceship-2-earth
if (isInside(spaceship.location, spaceship.size, earthLoc, earthSize.x)==true){
gameOver();
}
//spaceship-2-atmosphere
if (isInside(spaceship.location, spaceship.size, atmosphereLoc, atmosphereSize.x)==true){
spaceship.setNearEarth();
}
//bullet collisions
for (var i=0; i<asteroids.locations.length; i ){
for (var j=0; j<spaceship.bulletSys.bullets.length; j ){
if (isInside(spaceship.bulletSys.bullets[j], spaceship.bulletSys.diam, asteroids.locations[i], asteroids.diams[i]*2)==true){
asteroids.destroy(i);
}
}
}
}
//////////////////////////////////////////////////
//helper function checking if there's collision between object A and object B
function isInside(locA, sizeA, locB, sizeB){
var d = dist(locA.x, locA.y, locB.x, locB.y); //distance between both objects
var s = sizeA/2 sizeB/2; //sum of both objects' radius, size is divided by 2 since size is the diameter
if(d < s) //if distance between objects is smaller than sum of both objects' radius, overlap occurs
{
return true;
} else {
return false;
}
}
//////////////////////////////////////////////////
function keyPressed(){
if (keyIsPressed amp;amp; keyCode === 32){ // if spacebar is pressed, fire!
spaceship.fire();
}
}
//////////////////////////////////////////////////
// function that ends the game by stopping the loops and displaying "Game Over"
function gameOver(){
fill(255);
textFont(font);
textSize(80);
textAlign(CENTER);
text("GAME OVER", width/2, height/2)
noLoop();
}
//////////////////////////////////////////////////
// function that creates a star lit sky
function sky(){
push();
while (starLocs.length<300){
starLocs.push(new createVector(random(width), random(height)));
}
fill(255);
for (var i=0; i<starLocs.length; i ){
rect(starLocs[i].x, starLocs[i].y,2,2);
}
if (random(1)<0.3) starLocs.splice(int(random(starLocs.length)),1);
pop();
}
function preload() {
font = loadFont('assets/zorque.ttf');
asteroidimg = loadImage('assets/asteroid.png');
spaceshipimg = loadImage('assets/spaceship.png');
}
//////////////////////////////////////////////////
// function that keeps score of how many asteroids have been hit
function score(){
fill(255);
textFont(font);
textSize(30);
textAlign(CENTER);
text("Score: " asteroids.score, width/2, height-40)
}
Комментарии:
1. Loca или locb передаются как неопределенные
Ответ №1:
Это фрагмент кода, в котором произошла ошибка:
for (var i=0; i<asteroids.locations.length; i ){
for (var j=0; j<spaceship.bulletSys.bullets.length; j ){
if (isInside(spaceship.bulletSys.bullets[j], spaceship.bulletSys.diam, asteroids.locations[i], asteroids.diams[i]*2)){
asteroids.destroy(i);
}
}
}
Причина ошибки: при asteroids.destroy(i);
выполнении цикл все еще продолжает выполняться, if
проверка все еще выполняется. Если i == asteroids.locations.length - 1
это означает asteroids.locations[i]
, что это последний элемент в массиве. asteroids.destroy(i)
будет splice
ли это из массива, это все равно, что отрезать хвост массива. Тогда asteroids.locations[i]
будет равно undefined
. Он if
будет использовать его undefined
для проверки isInside
. isInside
примет locB.x
as undefined.x
и выдаст ошибку, которая Cannot read property 'x' of undefined
.
Предложите решение: вы должны break
выйти из for
цикла при уничтожении asteroids
. Например:
for (var i=0; i<asteroids.locations.length; i ){
for (var j=0; j<spaceship.bulletSys.bullets.length; j ){
if (isInside(spaceship.bulletSys.bullets[j], spaceship.bulletSys.diam, asteroids.locations[i], asteroids.diams[i]*2)){
asteroids.destroy(i);
break;
}
}
}