マウスストーカー

<script>

class P2{
  constructor(x=0,y=0){this.x=x;this.y=y}
  get clone(){return new P2(this.x,this.y)}//複写
  add({x=0,y=0},{x:X,y:Y}=this){this.x=X+x;this.y=Y+y;return this}//加算
  sub({x=0,y=0},{x:X,y:Y}=this){this.x=X-x;this.y=Y-y;return this}//減算
  mul({x=0,y=0},{x:X,y:Y}=this){this.x=X*x;this.y=Y*y;return this}//乗算
  div({x=0,y=0},{x:X,y:Y}=this){this.x=X/x;this.y=Y/y;return this}//除算
  sMul(n=0,{x:X,y:Y}=this){this.x=X*n;this.y=Y*n;return this}//スカラー倍
}
//__

//追いかける素となるもの(マウスの座標)
class MousePointer extends P2 {
  constructor (x, y) {
    super (x, y);
  }

  handleEvent (event) {
    this.x = event.clientX + window.pageXOffset;
    this.y = event.clientY + window.pageYOffset;
  }

  static create (view = document) {
    let obj = new this ();
    view.addEventListener ('mousemove', obj, false);
    return obj;
  }
}

//__
//追いかけるもの
class Stalker extends P2 {
  constructor (target, images, pointer, option = {}) {
    super (pointer.x, pointer.y);
    this.target   = target;
    this.images   = images;
    this.pointer  = pointer;
    this.option   = Object.assign ({}, this.constructor.getDefaultOption (), option);
    this.disabled = false;

    this.chase ();
  }


  chase () {
    const { PI, floor:int, atan2 } = Math;
    if (this.disabled) return;
    let
      dp = this.pointer.clone.sub (this),
      n = 360 / this.images.length,
      ang = atan2 (dp.x, -dp.y) * 180 / PI;

    this.add (dp.sMul (this.option.accelerator));
    this.locate ();
    this.target.src = this.images[int ((360 + ang) % 360 / n)].src;
    requestAnimationFrame (this.chase.bind (this));
  }


  locate () {
    const int = Math.floor;
    let p = this.option.offset.clone.add (this);
    Object.assign (this.target.style, { left: int (p.x) + 'px', top: int (p.y) + 'px'});
    return this;
  }


  static getDefaultOption () {
    return {
      offset: new P2 (0, 0),
      accelerator: .9
    };
  }


  static create (imageList, pointer, option = { }) {
    let
      target = new Image,
      images = imageList.map (src=> Object.assign (new Image, {src}));
    
    target.src = images[0].src;
    Object.assign (target.style, {position: 'absolute', left: '0px', top: '0px'});
    document.body.appendChild (target);

    return new Stalker (target, images, pointer, option);
  }
}

//__ 
  
const src = [
  '',
  '',
  '',
  '',
  '',
  '',
  '',
  ''
];

let
  pointer0 = MousePointer.create (),
  pointer1 = Stalker.create (src, pointer0, {offset: new P2(20,20), accelerator: .1}),
  pointer2 = Stalker.create (src, pointer1, {offset: new P2(40,40), accelerator: .09}),
  pointer3 = Stalker.create (src, pointer2, {offset: new P2(60,60), accelerator: .08});
  pointer4 = Stalker.create (src, pointer3, {offset: new P2(80,80), accelerator: .07});
  pointer5 = Stalker.create (src, pointer4, {offset: new P2(100,100), accelerator: .06});
  pointer6 = Stalker.create (src, pointer5, {offset: new P2(120,120), accelerator: .05});
  pointer7 = Stalker.create (src, pointer6, {offset: new P2(140,140), accelerator: .04});

</script>

```

```