CSS/Javascript 中的移动元素与静态元素相交

Moving element intersecting static element in CSS/Javascript

Video demo

我正在尝试使用 HTML、CSS 和 Javascript 为一个项目制作盗版 pacman(ish) 游戏。查看随附的视频以供参考。这个想法是:每当食物元素之一与 pacman 元素相交(即被吃掉)时,分数就会增加 1。每当食物元素离开视口而没有通过 pacman 元素时,玩家将失去 1 条生命。我尝试使用 Intersection Observer API 来实现这一点,方法是观察食物元素作为目标并将 pacman 元素作为根元素。但这没有用。

const obsCB1 = function (entries, observer) {
  point++;
  console.log(point);
};
const obsOptions1 = {
  root: pacman,
  threshold: 0.01,
};
const obs1 = new IntersectionObserver(obsCB1, obsOptions1);
obs1.observe(f1);

(食物元素最初从视口外向右开始,并在无限循环中使用 translateX 向左移动) 即使 f1 没有与 pacman 相交,回调也会在开始时被调用一次。而且这一点后来也没有改变。我怎样才能实现这个功能?

假设吃豆子和圆点都是圆,我想最好的办法是计算中心之间的距离并检查它是否小于半径之和

const board = document.querySelector('#board');
const score = board.querySelector('#score-val');
let dots = [];
function updateScore(x){score.innerText = +score.innerText+x}
function Dot(){
  this.SPEED = (.008 + Math.random()/1000) * window.innerHeight;
  this.RADIUS = .02 * window.innerHeight;
  this.el = document.createElement('span');
  this.el.style = `display: inline-block;width:${2*this.RADIUS}px;height:${2*this.RADIUS}px;position:absolute;left:${window.innerWidth}px;top:${Math.random()*(window.innerHeight-2*this.RADIUS)}px;border-radius:50%;background-color: red;`;
  board.appendChild(this.el);
  dots.push(this);
  this.remove = () => {
    dots = dots.filter(d => d!== this);
    this.el.remove();
  }
  this.move = () => {
    this.el.style.left = parseFloat(this.el.style.left)-this.SPEED+'px';
    if(pacmen.isCollapse(this)){
      this.remove();
      updateScore(1);
    } else 
    if(parseFloat(this.el.style.left)<0) { this.remove();
    updateScore(-1);
    }
  }
}
function Pacmen(){
  this.SPEED = .05 * window.innerHeight;
  this.RADIUS = .07 * window.innerHeight;
  this.el = board.querySelector('#pacmen');
  this.el.style.left=0;
  this.el.style.top=0;
  this.el.width=this.RADIUS*2;
  this.el.height=this.RADIUS*2;
  this.move = x => {
  this.el.style.top = Math.min(window.innerHeight-this.RADIUS*2, Math.max(0, parseFloat(this.el.style.top)+x*this.SPEED))+'px';
  }
  window.addEventListener('keydown', (e)=> {
  switch(e.key){
    case 'ArrowDown':
      e.preventDefault();
      this.move(1);
      break;
    case 'ArrowUp':
      e.preventDefault();
      this.move(-1);
      break;
  }
  });
  this.isCollapse = (dot) => {
    return Math.sqrt(Math.pow(parseFloat(this.el.style.left + this.RADIUS)-parseFloat(dot.el.style.left + dot.RADIUS),2)+Math.pow(parseFloat(this.el.style.top + this.RADIUS)-parseFloat(dot.el.style.top + dot.RADIUS),2)) <= this.RADIUS + dot.RADIUS
  }
}
const pacmen = new Pacmen();
function frame(){
  dots.forEach(dot =>dot.move());
  requestAnimationFrame(frame);
}
setInterval(()=>new Dot(), 1000);
frame();
html,body{padding:0;margin:0}
#board{
display:block;
width:100%;
height:100vh;
background-color:black;
position: relative;
overflow:hidden;
}
#pacmen {
  position: absolute;
}
#score{
  position:absolute;
  bottom: 0px;
  right: 20px;
  color: white;
  font-family: sans-serif;
}
<div id="board">
<img id="pacmen" src=""/>
<h3 id="score">Score: <span id="score-val">0</span><h3>
</div>