扭蛋机

Published on
/
/趣玩前端
<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <title>扭蛋机</title>
    <style>
      * {
        box-sizing: border-box;
        user-select: none;
      }
      body {
        padding: 0;
        margin: 0;
        font-family: sans-serif;
        background-color: #ff3636;
        overscroll-behavior: contain;
      }
      p,
      h1,
      h2,
      h3,
      h4 {
        display: inline-block;
        margin-block-start: 0em;
        margin-block-end: 0em;
        margin-inline-start: 0px;
        margin-inline-end: 0px;
        padding-inline-start: 0px;
      }
      .wrapper {
        position: absolute;
        width: 100%;
        height: 100%;
        overflow: hidden;
        display: flex;
        justify-content: center;
        align-items: center;
      }
      .pix {
        background-size: calc(var(--w) * var(--m)) calc(var(--h) * var(--m)) !important;
        background-repeat: no-repeat !important;
        image-rendering: pixelated;
      }
      .toy-box {
        position: absolute;
        top: -84px;
        width: 320px;
        height: 64px;
      }
      .capsule-machine {
        position: relative;
        width: 320px;
        height: 500px;
        --m: 4;
        --w: 80px;
        --h: 125px;
        background-image: url();
      }
      .capsule-machine.shake {
        animation: forwards shake 0.5s;
      }
      @keyframes shake {
        0%,
        40%,
        80% {
          margin-left: 5px;
        }
        20%,
        60%,
        100% {
          margin-left: -5px;
        }
      }
      .capsule-machine::after {
        position: absolute;
        content: '';
        width: calc(var(--w) * var(--m));
        height: calc(var(--h) * var(--m));
        background-size: calc(var(--w) * var(--m)) calc(var(--h) * var(--m)) !important;
        background-repeat: no-repeat !important;
        image-rendering: pixelated;
        background-image: url();
        pointer-events: none;
        z-index: 1;
        transition: 0.5s;
      }
      .capsule-machine.see-through::after,
      .capsule-machine.see-through .circle {
        opacity: 0.3;
      }
      .wrapper.lock {
        pointer-events: none;
      }
      .lock-cover {
        position: fixed;
        top: 0;
        left: 0;
        width: 100%;
        height: 100%;
        transition: 1s;
        background-color: transparent;
        pointer-events: none;
      }
      .wrapper.lock .lock-cover {
        background-color: #fab2cc;
        opacity: 0.8;
        z-index: 10;
      }
      .capsule {
        position: absolute;
        left: calc((var(--w) * var(--m)) / var(--offset));
        top: calc((var(--h) * var(--m)) / var(--offset));
        width: calc(var(--w) * var(--m));
        height: calc(var(--h) * var(--m));
        display: flex;
        align-items: center;
        justify-content: center;
      }
      .lid,
      .base {
        position: absolute;
        --h-m: var(--m) / 2;
        width: calc(var(--w) * var(--m));
        height: calc(var(--h) * var(--h-m));
        background-size: calc(var(--w) * var(--m)) calc(var(--h) * var(--h-m)) !important;
        background-repeat: no-repeat !important;
        image-rendering: pixelated;
      }
      .lid {
        background-image: url();
        top: 0;
        --start: 0;
        --end: -100px;
      }
      .base {
        background-image: url();
        bottom: 0;
        --start: 0;
        --end: 100px;
      }
      .base.white {
        background-image: url();
      }
      .base.pink {
        background-image: url();
      }
      .base.red {
        background-image: url();
      }
      .base.blue {
        background-image: url();
      }
      .capsule-wrapper.open .lid,
      .capsule-wrapper.open .base {
        animation: open-capsule forwards 1s;
      }
      @keyframes open-capsule {
        0% {
          transform: translateY(var(--start));
          opacity: 1;
        }
        100% {
          transform: translateY(var(--end));
          opacity: 0;
        }
      }
      .capsule-wrapper {
        position: absolute;
        --m: 4;
        --w: 16px;
        --h: 16px;
        --offset: -2;
        width: 0;
        height: 0;
        transition: 0.05s;
        cursor: pointer;
      }
      .capsule-wrapper.enlarge {
        --m: 8;
        z-index: 11;
        transition: 0.8s;
        pointer-events: none;
      }
      .capsule-wrapper.enlarge .toy {
        --m: 4;
      }
      .capsule-wrapper.enlarge.open .toy {
        --m: 6;
      }
      .capsule-wrapper.enlarge.open .toy.collected {
        --m: 2;
      }
      .line-start,
      .line-end {
        position: absolute;
        height: 0;
        width: 0;
      }
      .line {
        position: absolute;
        height: 1px;
        width: 0;
        border-top: solid #ff3636 4px;
      }
      .circle {
        position: absolute;
        bottom: 30px;
        left: 26px;
        width: 160px;
        height: 160px;
        display: flex;
        align-items: center;
        z-index: 5;
        background-image: url();
        --w: 40px;
        --h: 40px;
        --m: 4;
        transition: 0.5s;
      }
      .circle::after {
        position: absolute;
        top: -10px;
        right: -22px;
        content: '';
        --w: 15px;
        --h: 15px;
        --m: 4;
        width: calc(var(--w) * var(--m));
        height: calc(var(--h) * var(--m));
        background-image: url();
        background-size: calc(var(--w) * var(--m)) calc(var(--h) * var(--m)) !important;
        background-repeat: no-repeat !important;
        image-rendering: pixelated;
      }
      .handle {
        position: relative;
        width: 160px;
        height: 50px;
        display: flex;
        justify-content: space-between;
        cursor: pointer;
      }
      .handle::after {
        position: absolute;
        top: -7px;
        content: '';
        --w: 40px;
        --h: 16px;
        --m: 4;
        width: calc(var(--w) * var(--m));
        height: calc(var(--h) * var(--m));
        background-image: url();
        background-size: calc(var(--w) * var(--m)) calc(var(--h) * var(--m)) !important;
        background-repeat: no-repeat !important;
        image-rendering: pixelated;
      }
      .cover.white {
        position: absolute;
        bottom: 116px;
        width: 100%;
        height: calc(220px - 116px);
        background-color: white;
        z-index: 1;
        transition: 0.5s;
      }
      button {
        border: 0;
        background-color: white;
        padding: 4px 8px;
        font-family: 'Press Start 2P', sans-serif;
        font-size: 10px;
        margin-right: 10px;
      }
      .button-wrapper {
        position: absolute;
        left: 20px;
        bottom: 20px;
        z-index: 10;
      }
      .toy {
        position: absolute;
        --w: 16px;
        --h: 16px;
        --m: 2;
        width: calc(var(--w) * var(--m));
        height: calc(var(--h) * var(--m));
        z-index: -1;
        transition: 0.5s;
      }
      .bunny {
        background-image: url();
      }
      .duck-yellow {
        background-image: url();
      }
      .duck-pink {
        background-image: url();
      }
      .star {
        background-image: url();
      }
      .water-melon {
        background-image: url();
      }
      .panda {
        background-image: url();
      }
      .penguin {
        background-image: url();
      }
      .dino {
        background-image: url();
      }
      .roboto-san {
        background-image: url();
      }
      .roboto-sama {
        background-image: url();
      }
      .turtle {
        background-image: url();
      }
      .cover {
        position: absolute;
        z-index: 4;
        cursor: auto;
      }
      .cover.a {
        top: 0;
        left: 0;
        width: 320px;
        height: 280px;
      }
      .cover.b {
        top: 280px;
        left: 0;
        width: 212px;
        height: 220px;
      }
      .cover.c {
        top: 280px;
        left: 212px;
        width: 108px;
        height: 108px;
      }
      .cover.d {
        bottom: 0;
        left: 212px;
        width: 80px;
        height: 24px;
      }
      .cover.e {
        bottom: 0;
        left: 292px;
        height: 112px;
        width: 28px;
      }
      .sign {
        position: absolute;
        color: white;
        bottom: 10px;
        right: 10px;
        font-size: 10px;
      }
      a {
        color: white;
        text-decoration: none;
      }
      a:hover {
        text-decoration: underline;
      }
    </style>
  </head>
  <body>
    <body>
      <div class="wrapper">
        <div class="capsule-machine pix">
          <div class="toy-box"></div>
          <div class="lock-cover"></div>
          <div class="cover a"></div>
          <div class="cover b"></div>
          <div class="cover c"></div>
          <div class="cover d"></div>
          <div class="cover e"></div>
          <div class="circle pix"><div class="handle"></div></div>
        </div>
        <div class="button-wrapper">
          <button class="shake">摇一摇</button
          ><button class="see-inside">偷看</button>
        </div>
      </div>
    </body>
    <script>
      const settings = {
        capsuleNo: 20,
        isTurningHandle: false,
        isHandleLocked: false,
        handlePrevDeg: 0,
        handleDeg: 0,
        handleRotate: 0,
        flapRotate: 0,
        collectedNo: 0,
      };
      const elements = {
        wrapper: document.querySelector('.wrapper'),
        capsuleMachine: document.querySelector('.capsule-machine'),
        shakeButton: document.querySelector('.shake'),
        seeInsideButton: document.querySelector('.see-inside'),
        circle: document.querySelector('.circle'),
        handle: document.querySelector('.handle'),
        toyBox: document.querySelector('.toy-box'),
      };
      const vector = {
        x: 0,
        y: 0,
        create: function (x, y) {
          const obj = Object.create(this);
          obj.x = x;
          obj.y = y;
          return obj;
        },
        setXy: function ({ x, y }) {
          this.x = x;
          this.y = y;
        },
        setAngle: function (angle) {
          const length = this.magnitude();
          this.x = Math.cos(angle) * length;
          this.y = Math.sin(angle) * length;
        },
        setLength: function (length) {
          const angle = Math.atan2(this.y, this.x);
          this.x = Math.cos(angle) * length;
          this.y = Math.sin(angle) * length;
        },
        magnitude: function () {
          return Math.sqrt(this.x * this.x + this.y * this.y);
        },
        multiply: function (n) {
          return this.create(this.x * n, this.y * n);
        },
        addTo: function (v2) {
          this.x += v2.x;
          this.y += v2.y;
        },
        multiplyBy: function (n) {
          this.x *= n;
          this.y *= n;
        },
      };
      const rotatePoint = ({ angle, axis, point }) => {
        const a = degToRad(angle);
        const aX = point.x - axis.x;
        const aY = point.y - axis.y;
        return {
          x: aX * Math.cos(a) - aY * Math.sin(a) + axis.x,
          y: aX * Math.sin(a) + aY * Math.cos(a) + axis.y,
        };
      };
      const px = (num) => `${num}px`;
      const randomN = (max) => Math.ceil(Math.random() * max);
      const degToRad = (deg) => deg / (180 / Math.PI);
      const radToDeg = (rad) => Math.round(rad * (180 / Math.PI));
      const angleTo = ({ a, b }) => Math.atan2(b.y - a.y, b.x - a.x);
      const distanceBetween = (a, b) =>
        Math.round(Math.sqrt(Math.pow(a.x - b.x, 2) + Math.pow(a.y - b.y, 2)));
      const getPage = (e, type) =>
        e.type[0] === 'm' ? e[`page${type}`] : e.touches[0][`page${type}`];
      const calcCollectedX = () => (settings.collectedNo % 10) * 32;
      const calcCollectedY = () => Math.floor(settings.collectedNo / 10) * 32;
      const nearest360 = (n) =>
        n === 0 ? 0 : n - 1 + Math.abs(((n - 1) % 360) - 360);
      const setStyles = ({ el, x, y, w, deg }) => {
        if (w) el.style.width = w;
        el.style.transform = `translate(${x ? px(x) : 0},${
          y ? px(y) : 0
        })rotate(${deg || 0}deg)`;
      };
      const lineData = [
        {
          start: { x: 0, y: 280 },
          end: { x: 160, y: 360 },
          point: 'end',
          axis: 'start',
          id: 'flap_1',
        },
        {
          start: { x: 160, y: 360 },
          end: { x: 320, y: 280 },
          point: 'start',
          axis: 'end',
          id: 'flap_2',
        },
        {
          start: { x: 70, y: 340 },
          end: { x: 230, y: 490 },
          point: 'start',
          axis: 'end',
          id: 'ramp',
        },
      ];
      const getRandomToy = () => {
        return [
          'bunny',
          'duck-yellow',
          'duck-pink',
          'star',
          'water-melon',
          'panda',
          'dino',
          'roboto-san',
          'roboto-sama',
          'penguin',
          'turtle',
        ][randomN(11) - 1];
      };
      new Array(settings.capsuleNo).fill('').forEach(() => {
        const capsule = Object.assign(document.createElement('div'), {
          className: 'capsule-wrapper pix',
          innerHTML: `<div class="capsule"><div class="lid"></div><div class="${getRandomToy()} toy pix"></div><div class="base ${
            ['red', 'pink', 'white', 'blue'][randomN(4) - 1]
          }"></div></div>`,
        });
        elements.capsuleMachine.appendChild(capsule);
      });
      lineData.forEach(() => {
        [
          Object.assign(document.createElement('div'), {
            className: 'line-start',
            innerHTML: '<div class="line"></div>',
          }),
          Object.assign(document.createElement('div'), {
            className: 'line-end',
          }),
        ].forEach((ele) => {
          elements.capsuleMachine.appendChild(ele);
        });
      });
      const lineStarts = document.querySelectorAll('.line-start');
      const lines = document.querySelectorAll('.line');
      const lineEnds = document.querySelectorAll('.line-end');
      const toys = document.querySelectorAll('.toy');
      const { width: capsuleMachineWidth, height: capsuleMachineHeight } =
        elements.capsuleMachine.getBoundingClientRect();
      const handleAxis = () => {
        const { left: handleX, top: handleY } =
          elements.circle.getBoundingClientRect();
        const { top, left } = elements.capsuleMachine.getBoundingClientRect();
        return { x: handleX - left + 80, y: handleY - top + 80 };
      };
      const updateLines = () => {
        lineData.forEach((l, i) => {
          l.length = distanceBetween(l.start, l.end);
          setStyles({
            el: lineStarts[i],
            x: l.start.x,
            y: l.start.y,
            deg: radToDeg(angleTo({ a: l.start, b: l.end })),
          });
          setStyles({ el: lines[i], w: px(l.length) });
          setStyles({ el: lineEnds[i], x: l.end.x, y: l.end.y });
        });
      };
      const capsuleData = Array.from(
        document.querySelectorAll('.capsule-wrapper')
      ).map((c, i) => {
        const data = {
          ...vector,
          el: c,
          id: i,
          deg: 0,
          radius: 36,
          bounce: -0.3,
          friction: 0.99,
          toy: toys[i],
        };
        data.velocity = data.create(0, 1);
        data.velocity.setLength(10);
        data.velocity.setAngle(degToRad(90));
        data.setXy({
          x: randomN(capsuleMachineWidth - 32),
          y: randomN(capsuleMachineHeight - 250),
        });
        data.acceleration = data.create(0, 4);
        data.accelerate = function (acceleration) {
          this.velocity.addTo(acceleration);
        };
        return data;
      });
      const getNewPosBasedOnTarget = ({
        start,
        target,
        distance: d,
        fullDistance,
      }) => {
        const { x: aX, y: aY } = start;
        const { x: bX, y: bY } = target;
        const remainingD = fullDistance - d;
        return {
          x: Math.round((remainingD * aX + d * bX) / fullDistance),
          y: Math.round((remainingD * aY + d * bY) / fullDistance),
        };
      };
      const shake = () => {
        capsuleData.forEach((c) => {
          c.velocity.setAngle(degToRad(randomN(270)));
          c.velocity.setXy({ x: 10, y: 10 });
          c.accelerate(c.acceleration);
        });
        elements.capsuleMachine.classList.add('shake');
        setTimeout(
          () => elements.capsuleMachine.classList.remove('shake'),
          500
        );
      };
      const rotateLines = (angles) => {
        angles.forEach((angle, i) => {
          const { axis, point } = lineData[i];
          lineData[i][point] = rotatePoint({
            angle,
            axis: lineData[i][axis],
            point: lineData[i][point],
          });
        });
      };
      const openFlap = () => {
        if (settings.flapRotate > -20) {
          settings.flapRotate -= 2;
          rotateLines([2, -2, -4]);
          updateLines();
          setTimeout(openFlap, 30);
        } else {
          setTimeout(closeFlap, 800);
        }
      };
      const closeFlap = () => {
        if (settings.flapRotate < 0) {
          settings.flapRotate += 1;
          if (settings.flapRotate === 0) {
            [
              { x: 160, y: 360 },
              { x: 160, y: 360 },
              { x: 70, y: 340 },
            ].forEach((item, i) => {
              lineData[i][lineData[i].point].x = item.x;
              lineData[i][lineData[i].point].y = item.y;
            });
            settings.isHandleLocked = false;
          } else {
            rotateLines([-1, 1, 2]);
          }
          updateLines();
          setTimeout(closeFlap, 30);
        }
      };
      const release = () => {
        settings.flapRotate = 0;
        settings.isHandleLocked = true;
        setTimeout(openFlap, 30);
      };
      capsuleData.forEach((c) => {
        c.el.addEventListener('click', () => {
          const { width: bodyWidth, height: bodyHeight } =
            elements.wrapper.getBoundingClientRect();
          const { top, left } = elements.capsuleMachine.getBoundingClientRect();
          const { left: toyBoxLeft, top: toyBoxTop } =
            elements.toyBox.getBoundingClientRect();
          elements.wrapper.classList.add('lock');
          c.el.classList.add('enlarge');
          c.selected = true;
          setStyles({
            el: c.el,
            x: bodyWidth / 2 - left,
            y: bodyHeight / 2 - top,
            deg: nearest360(c.deg),
          });
          setStyles({ el: c.toy, deg: 0 });
          setTimeout(() => c.el.classList.add('open'), 700);
          setTimeout(() => {
            elements.wrapper.classList.remove('lock');
            c.toy.classList.add('collected');
            setStyles({
              el: c.el,
              x: toyBoxLeft - left + 16 + calcCollectedX(),
              y: toyBoxTop - top + 16 + calcCollectedY(),
            });
            settings.collectedNo++;
          }, 1800);
        });
        setStyles(c);
      });
      const spaceOutCapsules = (c) => {
        capsuleData.forEach((c2) => {
          if (c.id === c2.id || c2.selected) return;
          const distanceBetweenCapsules = distanceBetween(c, c2);
          if (distanceBetweenCapsules < c.radius * 2) {
            c.velocity.multiplyBy(-0.6);
            const overlap = distanceBetweenCapsules - c.radius * 2;
            c.setXy(
              getNewPosBasedOnTarget({
                start: c,
                target: c2,
                distance: overlap / 2,
                fullDistance: distanceBetweenCapsules,
              })
            );
          }
        });
      };
      const hitCheckLines = (c) => {
        lineData.forEach((l) => {
          const d1 = distanceBetween(c, l.start);
          const d2 = distanceBetween(c, l.end);
          if (
            d1 + d2 >= l.length - c.radius &&
            d1 + d2 <= l.length + c.radius
          ) {
            const dot =
              ((c.x - l.start.x) * (l.end.x - l.start.x) +
                (c.y - l.start.y) * (l.end.y - l.start.y)) /
              Math.pow(l.length, 2);
            const closestXy = {
              x: l.start.x + dot * (l.end.x - l.start.x),
              y: l.start.y + dot * (l.end.y - l.start.y),
            };
            const fullDistance = distanceBetween(c, closestXy);
            if (fullDistance < c.radius) {
              c.velocity.multiplyBy(-0.6);
              const overlap = fullDistance - c.radius;
              c.setXy(
                getNewPosBasedOnTarget({
                  start: c,
                  target: closestXy,
                  distance: overlap / 2,
                  fullDistance,
                })
              );
            }
          }
        });
      };
      const hitCheckCapsuleMachineWalls = (c) => {
        const buffer = 5;
        if (c.x + c.radius + buffer > capsuleMachineWidth) {
          c.x = capsuleMachineWidth - (c.radius + buffer);
          c.velocity.x = c.velocity.x * c.bounce;
        }
        if (c.x - (c.radius + buffer) < 0) {
          c.x = c.radius;
          c.velocity.x = c.velocity.x * c.bounce;
        }
        if (c.y + c.radius + buffer > capsuleMachineHeight) {
          c.y = capsuleMachineHeight - c.radius - buffer;
          c.velocity.y = c.velocity.y * c.bounce;
        }
        if (c.y - c.radius < 0) {
          c.y = c.radius;
          c.velocity.y = c.velocity.y * c.bounce;
        }
      };
      const animateCapsules = () => {
        capsuleData.forEach((c, i) => {
          if (c.selected) return;
          c.prevX = c.x;
          c.prevY = c.y;
          c.accelerate(c.acceleration);
          c.velocity.multiplyBy(c.friction);
          c.addTo(c.velocity);
          spaceOutCapsules(c);
          hitCheckLines(c);
          hitCheckCapsuleMachineWalls(c);
          if (Math.abs(c.prevX - c.x) < 2 && Math.abs(c.prevY - c.y) < 2) {
            c.velocity.setXy({ x: 0, y: 0 });
            c.setXy({ x: c.prevX, y: c.prevY });
          } else {
            if (Math.abs(c.prevX - c.x)) {
              setStyles({ el: c.toy, deg: c.deg + (c.x - c.prevX) * 2 });
              c.deg += (c.x - c.prevX) * 2;
            }
          }
          setStyles(capsuleData[i]);
        });
      };
      const grabHandle = (e) => {
        if (settings.isHandleLocked) return;
        const { top, left } = elements.capsuleMachine.getBoundingClientRect();
        settings.isTurningHandle = true;
        settings.handleDeg = radToDeg(
          angleTo({
            a: { x: getPage(e, 'X') - left, y: getPage(e, 'Y') - top },
            b: handleAxis(),
          })
        );
        settings.handleRotate = 0;
      };
      const releaseHandle = () => {
        settings.isTurningHandle = false;
        setStyles({ el: elements.handle, deg: 0 });
      };
      const rotateHandle = (e) => {
        if (!settings.isTurningHandle || settings.isHandleLocked) return;
        const { top, left } = elements.capsuleMachine.getBoundingClientRect();
        settings.prevHandleDeg = settings.handleDeg;
        const deg = radToDeg(
          angleTo({
            a: { x: getPage(e, 'X') - left, y: getPage(e, 'Y') - top },
            b: handleAxis(),
          })
        );
        settings.handleDeg = deg;
        const diff = settings.handleDeg - settings.prevHandleDeg;
        if (diff >= 1) {
          setStyles({ el: elements.handle, deg: settings.handleRotate });
        }
        if (diff > 0 && diff < 50) settings.handleRotate += diff;
        if (settings.handleRotate > 350) {
          setStyles({ el: elements.handle, deg: 10 });
          release();
          settings.isTurningHandle = false;
        }
      };
      ['mousedown', 'touchstart'].forEach((action) => {
        elements.handle.addEventListener(action, grabHandle);
      });
      ['mouseup', 'mouseleave', 'touchend'].forEach((action) => {
        elements.circle.addEventListener(action, releaseHandle);
      });
      ['mousemove', 'touchmove'].forEach((action) => {
        window.addEventListener(action, rotateHandle);
      });
      elements.shakeButton.addEventListener('click', shake);
      elements.seeInsideButton.addEventListener('click', () => {
        elements.capsuleMachine.classList.toggle('see-through');
        elements.seeInsideButton.innerHTML =
          elements.capsuleMachine.classList.contains('see-through')
            ? '隐藏'
            : '偷看';
      });
      updateLines();
      setInterval(animateCapsules, 30);
    </script>
  </body>
</html>