p5.j​​s 中的碰撞处理

2023-11-24

我一直在使用 p5.js,并且已经弄清楚如何让对象来检测碰撞,但我对如何处理这些碰撞感到非常困惑。我尝试将玩家速度设置为 0,但随后玩家卡住了。我也尝试过将碰撞分开到每一侧,但这也不起作用。

这是我的带有碰撞函数的玩家构造函数:

class Player {

    constructor() {
        this.w = 50;
        this.h = 125;
        this.pos = createVector(0, 0);
        this.vel = createVector();
        this.acc = acceleration;
        this.grav = gravity;
    }

    update() {
      this.vel.y += this.grav;
      this.pos.y += this.vel.y;

      this.vel.x += this.acc;

      this.pos.y = constrain(this.pos.y, 0, canvasHeight - this.h);
      this.pos.x = constrain(this.pos.x, 0, canvasWidth - this.w);
      image(playerImg, this.pos.x, this.pos.y, this.w, this.h);
    }

    run() {
      this.update();
    } 

    isOnFloor() {
      return(this.pos.y >= canvasHeight - this.h);
    }

    collides(x, y, w, h) {
      if (this.pos.x >= x - this.w && this.pos.x <= x+w && 
      this.pos.y >= y - this.h && this.pos.y <= y+h) {
        return true;
      } else {
        return false;
      }
    }
    collidesY(y, h) {
      if (this.pos.y >= y - this.h && this.pos.y <= y+h) {
        return true;
      } else {
        return false;
      }
    }
    collidesX(x, w) {
      if (this.pos.x >= x - this.w && this.pos.x <= x+w) {
        return true;
      } else {
        return false;
      }
    }
    collidesXL(x) { //X axis Left
      if (this.pos.x >= x - this.w) {
        return true;
      } else {
        return false;
      }
    }
    collidesXR(x, w) { //X axis Right
      if (this.pos.x <= x+w) {
        return true;
      } else {
        return false;
      }
    }
    collidesYT(y, h) { //Y axis Top
      if (this.pos.y <= y+h) {
        return true;
      } else {
        return false;
      }
    }
    collidesYB(y) { //Y axis Bottom
      if (this.pos.y >= y - this.h) {
        return true;
      } else {
        return false;
      }
    }
    
}

正如您所看到的,我创建了许多函数来尝试使碰撞处理正常工作。这是我的主要代码,第 62 行有碰撞处理。第 54 行显示了我已注释掉的代码,因为它不起作用。

var canvasWidth = window.innerWidth
var canvasHeight = window.innerHeight

let player;
var playerImg;

let platform;

let speed = 5;
let acceleration = 0.075;
let jumpForce = -10;
let gravity = 0.25;

let velocityDebug = false; //keep this off because of lag

function setup() {
  createCanvas(canvasWidth, canvasHeight)

  player = new Player()
  block = new Block(canvasWidth / 3, canvasHeight / 1.2, 200, 25)

}

function preload() {
  playerImg = loadImage(palyerImageData);
}

function draw() {

  background(50, 200, 255)

  player.run();
  block.display();

  if (keyIsPressed && keyCode === UP_ARROW && player.isOnFloor()) {
    player.vel.y = jumpForce;
  } else if (keyIsPressed && keyCode === RIGHT_ARROW) {
    player.pos.x += player.vel.x;
  } else if (keyIsPressed && keyCode === LEFT_ARROW) {
    player.pos.x -= player.vel.x;
  } else {
    player.vel.x = speed;
  }

  function keyReleased() {
    player.vel.x = speed;
  }

  /*if (player.collides(block.x, block.y, block.w, block.h)) {
      player.vel.y = 0;
      player.vel.x = 0;
      player.grav = 0;
    } else {
      player.grav = gravity;
    }*/

  if (player.collides(block.x, block.y, block.w, block.h)) {

    if (player.collidesXL(block.x)) {

    }
    if (player.collidesXR(block.x, block.w)) {

    }
    if (player.collidesYT(block.y, block.h)) {

    }
    if (player.collidesYB(block.y)) {

    }
  } else {
    player.acc = acceleration;
    player.grav = gravity;
  }

  if (velocityDebug) {
    console.log("Y Vel: " + player.vel.y)
    console.log("X Vel:" + player.vel.x)
  }

}


// Player class (same as above)
class Player {
  constructor() {
    this.w = 50;
    this.h = 125;
    this.pos = createVector(0, 0);
    this.vel = createVector();
    this.acc = acceleration;
    this.grav = gravity;
  }

  update() {
    this.vel.y += this.grav;
    this.pos.y += this.vel.y;

    this.vel.x += this.acc;

    this.pos.y = constrain(this.pos.y, 0, canvasHeight - this.h);
    this.pos.x = constrain(this.pos.x, 0, canvasWidth - this.w);
    image(playerImg, this.pos.x, this.pos.y, this.w, this.h);
  }

  run() {
    this.update();
  }

  isOnFloor() {
    return (this.pos.y >= canvasHeight - this.h);
  }

  collides(x, y, w, h) {
    if (this.pos.x >= x - this.w && this.pos.x <= x + w &&
      this.pos.y >= y - this.h && this.pos.y <= y + h) {
      return true;
    } else {
      return false;
    }
  }
  collidesY(y, h) {
    if (this.pos.y >= y - this.h && this.pos.y <= y + h) {
      return true;
    } else {
      return false;
    }
  }
  collidesX(x, w) {
    if (this.pos.x >= x - this.w && this.pos.x <= x + w) {
      return true;
    } else {
      return false;
    }
  }
  collidesXL(x) { //X axis Left
    if (this.pos.x >= x - this.w) {
      return true;
    } else {
      return false;
    }
  }
  collidesXR(x, w) { //X axis Right
    if (this.pos.x <= x + w) {
      return true;
    } else {
      return false;
    }
  }
  collidesYT(y, h) { //Y axis Top
    if (this.pos.y <= y + h) {
      return true;
    } else {
      return false;
    }
  }
  collidesYB(y) { //Y axis Bottom
    if (this.pos.y >= y - this.h) {
      return true;
    } else {
      return false;
    }
  }
}

class Block {
  constructor(x, y, w, h) {
    this.x = x;
    this.y = y;
    this.w = w;
    this.h = h;
    this.color = color(225, 225, 255)
  }
  display() {
    fill(this.color)
    rect(this.x, this.y, this.w, this.h)
  }
}

const palyerImageData = "";
body, html {
margin: 0;
width: 100%;
height: 100%;
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.4.0/p5.js"></script>

我已经为此工作了数周,并且做了大量测试,但没有任何效果。如果不将玩家速度更改为 0,我似乎无法阻止玩家从平台上掉下来,这当然会使玩家无法移动。请帮忙。我应该怎么做才能解决这个问题?

以下是 replit 的完整代码,以及您可以尝试的演示:

Code: https://replit.com/@STCollier/Player-Movement-Script?v=1 Demo: https://player-movement-script.stcollier.repl.co/

太感谢了!


这有点复杂,如果有更好的解决方案(无论是在优雅还是性能方面)我都不会感到惊讶。看来您需要考虑几种不同的碰撞场景。

Collision Diagram

在绿色的情况下,情况很简单,玩家的一侧边缘与方块的一侧边缘发生碰撞。检查这一点涉及确认玩家的边缘与块重叠,并且玩家的两个角都在块内或两个角都在块外。这些情况在图中为绿色。在这些情况下,您可以简单地约束水平碰撞的 X 位置和垂直碰撞的 Y 位置。

复杂的情况是当玩家与角落的方块碰撞时。在这种情况下,您需要弄清楚碰撞是首先发生在侧面还是顶部/底部。这可以通过使用两个角之间的水平距离乘以斜率并将所得位置与块的相应角进行比较来完成。

Corner Collision Diagram

在这个例子中,因为玩家的右下角,当在运动方向上偏移到碰撞点时是below块的左上角,玩家的水平位置应该受到限制(即他们跑到块的侧面而不是顶部)。

下面是示例代码(放慢速度并突出显示,以便更容易看到正在发生的情况):

var canvasWidth = window.innerWidth
var canvasHeight = window.innerHeight

let player;
var playerImg;

let platform;

let speed = 5;
let acceleration = 0.075;
let jumpForce = -10;
let gravity = 0.25;

function setup() {
  createCanvas(canvasWidth, canvasHeight)

  player = new Player()
  block = new Block(canvasWidth / 3, canvasHeight / 1.2, 200, 25)
  frameRate(15);
}

function preload() {
  playerImg = loadImage(palyerImageData);
}

let first = true;
function draw() {
  background(50, 200, 255, 20)
 
  let initX = player.pos.x;
  let initY = player.pos.y;
 
  player.update();
  block.display();

  if (keyIsPressed && keyCode === UP_ARROW && player.isOnFloor()) {
    player.vel.y = jumpForce;
  } else if (keyIsPressed && keyCode === RIGHT_ARROW) {
    player.pos.x += player.vel.x;
  } else if (keyIsPressed && keyCode === LEFT_ARROW) {
    player.pos.x -= player.vel.x;
  } else {
    player.vel.x = speed;
  }
  
  // Calculate the direction of movement so we no what edges to detect collisions with.
  let deltaX = player.pos.x - initX;
  let deltaY = player.pos.y - initY;
  
  // This doesn't do anything here. It would need to be declared globally
  /*
  function keyReleased() {
    player.vel.x = speed;
  } */

  if (player.collides(block.x, block.y, block.w, block.h)) {
    let dir = '';
    if (deltaY < 0) dir += 'N';
    else if (deltaY > 0) dir += 'S';

    if (deltaX < 0) dir += 'W';
    else if (deltaX > 0) dir += 'E';

    let tr = block.inside(player.pos.x + player.w, player.pos.y);
    let br = block.inside(player.pos.x + player.w, player.pos.y + player.h);
    let tl = block.inside(player.pos.x, player.pos.y);
    let bl = block.inside(player.pos.x, player.pos.y + player.h);
    push();
    stroke('red');
    strokeWeight(2);
    if (dir === 'E' ||
        deltaX > 0 && player.collidesY(block.y, block.h) &&
        // The player's right side collided with the block's left side
        player.collidesXL(block.x, block.w) &&
        // And either both the top right and bottom right of the player are inside the block or neither of them are.
        xnor(tr, br)) {
      // simple case: our right edge collided with the block's left edge
      player.pos.x = block.x - player.w - 1;
      line(block.x, block.y, block.x, block.y + block.h);
    } else if (dir === 'W' ||
               deltaX < 0 && player.collidesY(block.y, block.h) && player.collidesXR(block.x, block.w) && xnor(tl, bl)) {
      // simple case: our left edge collided with the block's right edge
      player.pos.x = block.x + block.w + 1;
      line(block.x + block.w, block.y, block.x + block.w, block.y + block.h);
    } else if (dir === 'S' ||
               deltaY > 0 && player.collidesX(block.x, block.w) && player.collidesYB(block.y, block.h) && xnor(bl, br)) {
      // simple case: our bottom edge collided with the block's top edge
      player.pos.y = block.y - player.h - 1;
      // reset velocity due to gravity
      if (player.vel.y > 0) {
        player.vel.y = player.grav;
      }
      line(block.x, block.y, block.x + block.w, block.y);
    } else if (dir === 'N' ||
               deltaY < 0 && player.collidesX(block.x, block.w) && player.collidesYT(block.y, block.h) && xnor(tl, tr)) {
      // simple case: our top edge collided with the block's bottom edge
      player.pos.y = block.y + block.h + 1;
      line(block.x, block.y + block.h, block.x + block.w, block.y + block.h);
    } else {
      // nasty case, we collided from a corner, we need to determine whether to constrain the player's position in the X or in the Y direction.
      
      let slope = deltaY / deltaX;
      stroke('limegreen');
      strokeWeight(6);
      // At this point only one corner should be inside the box
      if (tr) {
        // At the point where the player's Top Right intersected the block, was the block's Bottom Left above or below the player's Top Right
        if (initX + player.w >= block.x || initY + (block.x - (initX + player.w)) * slope > block.y + block.h) {
          // Since the players TR was below the block's BL at the time of collision, constrain the Player's Y position
          player.pos.y = block.y + block.h + 1;
        } else {
          player.pos.x = block.x - player.w - 1;
        }
        point(block.x, block.y + block.h);
      } else if (br) {
        if (initX + player.w >= block.x || initY + player.h + (block.x - (initX + player.w)) * slope < block.y) {
          player.pos.y = block.y - player.h - 1;
          // reset velocity due to gravity
          if (player.vel.y > 0) {
            player.vel.y = player.grav;
          }
        } else {
          player.pos.x = block.x - player.w - 1;
        }
        point(block.x, block.y);
      } else if (bl) {
        if (initX <= block.x + block.w ||
            initY + player.h + (initX - (block.x + block.w)) * slope < block.y) {
          player.pos.y = block.y - player.h - 1;
          // reset velocity due to gravity
          if (player.vel.y > 0) {
            player.vel.y = player.grav;
          }
        } else {
          player.pos.x = block.x + block.w + 1;
        }
        point(block.x + block.w, block.y);
      } else if (tl) {
        if (initX <= block.x + block.w ||
            initY + (initX - (block.x + block.w)) * slope > block.y + block.h) {
          player.pos.y = block.y - player.h - 1;
        } else {
          player.pos.x = block.x + block.w + 1;
        }
        point(block.x + block.w, block.y + block.h);
      }
    }
    pop();
  }
  
  player.display();
}

function xnor(a, b) {
  return (a && b) || (!a && !b);
}


// Player class (same as above)
class Player {
  constructor() {
    this.w = 50;
    this.h = 125;
    this.pos = createVector(0, 0);
    this.vel = createVector();
    this.acc = acceleration;
    this.grav = gravity;
  }

  update() {
    this.vel.y += this.grav;
    this.pos.y += this.vel.y;

    this.vel.x += this.acc;

    this.pos.y = constrain(this.pos.y, 0, canvasHeight - this.h);
    this.pos.x = constrain(this.pos.x, 0, canvasWidth - this.w);
  }

  display() {
    image(playerImg, this.pos.x, this.pos.y, this.w, this.h);
  }

  isOnFloor() {
    return (this.pos.y >= canvasHeight - this.h);
  }

  collides(x, y, w, h) {
    if (this.pos.x + this.w >= x && this.pos.x <= x + w &&
      this.pos.y + this.h >= y && this.pos.y <= y + h) {
      return true;
    } else {
      return false;
    }
  }
  collidesY(y, h) {
    if (this.pos.y >= y - this.h && this.pos.y <= y + h) {
      return true;
    } else {
      return false;
    }
  }
  collidesX(x, w) {
    if (this.pos.x >= x - this.w && this.pos.x <= x + w) {
      return true;
    } else {
      return false;
    }
  }
  // Check if the right side of the player is inside the box
  collidesXL(x, w) { //X axis Left
    if (this.pos.x + this.w >= x && this.pos.x + this.w <= x + w) {
      return true;
    } else {
      return false;
    }
  }
  // Check if the left side of the player is inside the box
  collidesXR(x, w) { //X axis Right
    if (this.pos.x <= x + w && this.pos.x >= x) {
      return true;
    } else {
      return false;
    }
  }
  // Check if the top side of the player is inside the box
  collidesYT(y, h) { //Y axis Top
    if (this.pos.y <= y + h && this.pos.y >= y) {
      return true;
    } else {
      return false;
    }
  }
  // Check if the bottom side of the player is inside the box
  collidesYB(y, h) { //Y axis Bottom
    if (this.pos.y + this.h >= y && this.pos.y <= y + h) {
      return true;
    } else {
      return false;
    }
  }
}

class Block {
  constructor(x, y, w, h) {
    this.x = x;
    this.y = y;
    this.w = w;
    this.h = h;
    this.color = color(225, 225, 255, 20)
  }
  display() {
    fill(this.color)
    stroke(0, 0, 0, 50);
    rect(this.x, this.y, this.w, this.h)
  }
  inside(x, y) {
    return x >= this.x && x <= this.x + this.w &&
      y >= this.y && y <= this.y + this.h;
  }
}

const palyerImageData = "";
body, html {
margin: 0;
width: 100%;
height: 100%;
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.4.0/p5.js"></script>
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

p5.j​​s 中的碰撞处理 的相关文章

随机推荐

  • 使用 RestSharp 发布数据时序列化对象

    我最近开始使用 RestSharp 来使用使用 XML 的 REST 服务 它使得将 XML 对象反序列化为自定义对象集合变得非常简单 但我的问题是最好的方法是什么重新序列化什么时候发回服务 我应该使用 LINQ to XML 重新序列化吗
  • 如何以编程方式将 mp3 转换为 itunes 可播放的 aac/m4a 文件?

    我一直在寻找一种以编程方式或通过命令行将 mp3 转换为 aac 的方法 但没有成功 理想情况下 我有一段代码可以从我的 Rails 应用程序中调用 将 mp3 转换为 aac 我安装了 ffmpeg 和 libfaac 并能够使用以下命令
  • 大O符号之和[重复]

    这个问题在这里已经有答案了 可能的重复 将不同的例程组合在一起时的大O 什么是O n O log n 减少到 我的猜测是O n 但无法给出严格的推理 我明白O n O 1 应该减少到O n since O 1 只是一个常数 好吧 自从O f
  • 使用 ClosedXML 将 Gridview 导出到 Excel,而不发出警告:您尝试打开的文件格式不同

    我正在开发 ASP NET 4 5 Webform 并且有一个 Gridview 具有自定义 TemplateField 并从 sqlDataSource 获取数据 我有这个事件将 gridview 内容导出到 Excel 工作表 它的工作
  • 如何从 C# 中的一组图像创建 H.264 编码的电影?

    我今天做了很多搜索 C 库 这将允许我创建 H 264 编码的视频文件 有谁知道是否存在这样的库或第三方组件 Use ffmpeg 我建议直接运行 ffmpeg 进程 如本示例所示 http jasonjano wordpress com
  • 递归释放 C 结构体

    我有一个结构 仅包含指向我分配的内存的指针 有没有一种方法可以递归地释放每个作为指针的元素 而不是对每个元素调用 free 例如 假设我有这样的布局 typedef struct vertex typedef struct normal t
  • 如何将 SQL 数据库嵌入/附加到 Visual C# 中?

    这是我第一次使用 SQL 这可能是一个愚蠢的问题 但我做了一些研究 但我认为我没有找到我想要的东西 我想要的是一种创建将在我的 C 程序中使用的私有 SQL 数据库的方法 我已经在 SQL Server Express 中创建了一个数据库
  • C++中虚表的结构是怎样的?

    例如 我有两个 接口 和类类型 class IPlugin public virtual void Load void 0 virtual void Free void 0 class IFoo public virtual void Fo
  • 在哪里下载 your_client_secret_File.json 文件

    我正在使用 YouTube API 但首先 我需要凭证文件 your client secret File json 通过遵循本教程https developers google com youtube analytics referenc
  • Clang 中 __int128_t 的错误?

    这段小代码可以使用 GCC 和 Clang 进行编译 但会给出不同的结果 include
  • Python Twisted 中的内存泄漏:它在哪里?

    我有一个负载下的 Twisted 服务器 当服务器处于负载状态时 内存使用量会增加 并且永远不会被回收 即使没有更多的客户端 下次进入高负载时 内存使用量会再次增加 这是当时情况的快照 RSS 内存为 400 MB 通常最大客户端数应为 2
  • HTML5 验证中的空标题警告

    我尝试验证我的 HTML 代码 但发现了此警告 空标题 See here 显然标题不为空 的内容 h3 h3 在 WordPress 中由以下方式生成the title 功能 所以 我不明白为什么会发生这种情况 有人可以解释一下问题是什么吗
  • 使用接口进行 IList 和 List 转换

    我大致了解接口 继承和多态性 但有一件事让我感到困惑 在这个例子中 Cat实施IAnimal而且当然List实施IList IList
  • 如何在 html 游戏中保存进度

    我想知道如何保存玩家在我正在制作的游戏中取得的进度 我可以通过 cookies 来做到这一点吗 或者我还可以如何将其保存到玩家的计算机上 感谢您的所有帮助 使用 Javascript 在本地保存几乎有两种选择 它们是cookies and
  • 将 Eigen 库添加到 Android NDK

    我需要在我的 Android Studio 项目中包含 Eigen 库来执行一些线性代数运算并使用我为桌面开发的一些 C 代码 我一直在寻找有关此主题的信息 但没有太多 而且我是 Android NDK 的新手 Eigen 库不需要编译 所
  • 正则表达式匹配除字符列表之外的所有内容

    我想匹配包含除指定字符之外的所有内容的行 I V X M C D L new Regex is I V X M C D L 应该匹配除中提到的字符之外的所有内容OR list 应匹配 name is a 不应该匹配 edition is I
  • Python IMAP 搜索指定电子邮件地址或发送至指定电子邮件地址

    我将其与 Gmail 的 SMTP 服务器一起使用 并且我想通过 IMAP 搜索发送到某个地址或从某个地址接收的电子邮件 这就是我所拥有的 mail imaplib IMAP4 SSL imap gmail com mail login u
  • 命名约定 - C++ 和 C# 变量中的下划线

    常见的是看到一个 var类字段中的变量名 下划线是什么意思 所有这些特殊命名约定都有参考吗 下划线只是一种约定 而已 因此 每个人的使用总是有些不同 以下是我对这两种语言的理解 在C 中 下划线通常表示私有成员变量 在 C 中 我通常只在为
  • 使用绝对定位是否被认为是不好的做法? [关闭]

    Closed 这个问题是基于意见的 目前不接受答案 我正在开发一个网页 在其中放置一个类似国际象棋的游戏的棋盘 以及几个棋子托盘 这一切都是使用 HTML 完成的 使用 jQuery 在玩游戏时进行动态更新 在某个地方 我认为在页面内使用元
  • p5.j​​s 中的碰撞处理

    我一直在使用 p5 js 并且已经弄清楚如何让对象来检测碰撞 但我对如何处理这些碰撞感到非常困惑 我尝试将玩家速度设置为 0 但随后玩家卡住了 我也尝试过将碰撞分开到每一侧 但这也不起作用 这是我的带有碰撞函数的玩家构造函数 class P