Pen Settings

HTML

CSS

CSS Base

Vendor Prefixing

Add External Stylesheets/Pens

Any URLs added here will be added as <link>s in order, and before the CSS in the editor. You can use the CSS from another Pen by using its URL and the proper URL extension.

+ add another resource

JavaScript

Babel includes JSX processing.

Add External Scripts/Pens

Any URL's added here will be added as <script>s in order, and run before the JavaScript in the editor. You can use the URL of any other Pen and it will include the JavaScript from that Pen.

+ add another resource

Packages

Add Packages

Search for and use JavaScript packages from npm here. By selecting a package, an import statement will be added to the top of the JavaScript editor for this package.

Behavior

Auto Save

If active, Pens will autosave every 30 seconds after being saved once.

Auto-Updating Preview

If enabled, the preview panel updates automatically as you code. If disabled, use the "Run" button to update.

Format on Save

If enabled, your code will be formatted when you actively save your Pen. Note: your code becomes un-folded during formatting.

Editor Settings

Code Indentation

Want to change your Syntax Highlighting theme, Fonts and more?

Visit your global Editor Settings.

HTML

              
                <style>
    @import url('https://fonts.googleapis.com/css2?family=Press+Start+2P&display=swap');
</style>
<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">shake</button>
      <button class="see-inside">see inside</button>
    </div>

    <div class="sign">
      by masahito / <a href="http://www.ma5a.com/">ma5a.com</a>
    </div>
  </div>
</body>
              
            
!

CSS

              
                
* {
  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;
/*   padding-top: 130px; */
}

.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();
}
/* #2cbee2 */

.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 {
  /* border: 1px solid hotpink; */
  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;
}

              
            
!

JS

              
                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, // actual radius should be 32, but setting it higher
      bounce: -0.3, // this reduces the velocity gradually
      friction: 0.99,
      toy: toys[i]
    }

    data.velocity = data.create(0, 1)  //? velocity is another vector
    data.velocity.setLength(10)
    data.velocity.setAngle(degToRad(90))
    data.setXy({
      x: randomN(capsuleMachineWidth - 32), 
      y: randomN(capsuleMachineHeight - 250), 
    })

    // gravity
    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)) {
          // rotate capsule
          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') ? 'hide' : 'see inside'
  })

  updateLines()
  setInterval(animateCapsules, 30)

              
            
!
999px

Console