WEBLEB
Home
Editor
Accedi
Pro
Italiano
English
Français
Español
Português
Deutsch
Italiano
हिंदी
macchina artiglio
564
zegarkidawida
Apri nell'Editor
Pubblica il Tuo Codice
HTML
Copy
by masahito /
ma5a.com
CSS
Copy
*, ::before, ::after { box-sizing: border-box; } body { padding: 0; margin: 0; font-family: sans-serif; background-color: #84dfe2; --brown: #57280f; --blue: #33a5da; --dark-blue: #3a94b7; --machine-color: #32c2db; --machine-width: 160px; --m: 2; } .wrapper { display: flex; justify-content: center; align-items: center; height: 100vh; min-height: 600px; flex-direction: column; } .pix, .pix::before, .pix::after { width: calc(var(--w) * var(--m)); height: calc(var(--h) * var(--m)); image-rendering: pixelated; background-size: calc(var(--w) * var(--m)) calc(var(--h) * var(--m)); } .pix::before, .pix::after { position: absolute; content: ''; } .claw-machine { display: flex; flex-direction: column; border: 2px solid var(--brown); overflow: hidden; } .collection-box { --h: 32px; width: calc(var(--m) * var(--machine-width)); margin: calc(var(--m) * var(--h) * -1 - 10px) 0 24px; display: flex; align-items: bottom; justify-content: start; } .collection-box .toy-wrapper { position: relative; width: calc(100% / 6); display: flex; align-items: end; justify-content: center; } .collection-box .toy-wrapper .toy::after { top: auto; bottom: 0; } .toy-wrapper.squeeze-in { width: 0; animation: squeeze-in forwards 0.4s; animation-delay: 1.4s; } @keyframes squeeze-in { 0% { width: 0; } 100% { width: calc(100% / 6); } } @keyframes show-toy-2 { 0% { opacity: 0; transform: translateY(-100vh); } 100% { opacity: 1; transform: translateY(0); } } .toy-wrapper .toy { opacity: 0; transform: translateY(-100vh); animation: forwards show-toy-2 0.8s; animation-delay: 1s; } .claw-machine.show-overlay::after { content: ''; position: absolute; width: 100%; height: 100%; left: 0; top: 0; background-color: var(--machine-color); opacity: 0.6; z-index: 6; pointer-events: none; } .box { position: relative; background-color: #7fcfed; --w: var(--machine-width); --h: 180px; } .box::before { background-color: var(--brown); top: calc(70px * var(--m)); width: 100%; height: calc(var(--m) * 8px); z-index: 5; } .box::after { background-color: var(--machine-color); top: calc(70px * var(--m)); right: 0px; width: calc(var(--m) * 8px); height: calc(var(--m) * 170px); z-index: 2; } .machine-top, .machine-bottom { position: absolute; width: 100%; --h: 70px; height: calc(var(--h) * var(--m)); } .machine-top { top: 0; display: flex; align-items: center; z-index: 1; } .machine-top::after { content: ''; top: 0; left: 0; width: 100%; height: 100%; background-color: #70f7f372; z-index: 1; } .machine-bottom { bottom: 0; background-color: #def7f6; } .collection-point { position: absolute; bottom: 0; background-image: url(); --w: 44px; --h: 24px; z-index: 0; } .rail { top: 0; left: 0; transition: 0.3s; border: solid var(--blue); } .rail.hori { --h: 5px; width: 100%; background-color: #fff; border-width: 2px 0; z-index: 1; } .rail.vert { position: absolute; --h: 70px; --w: 5px; background-color: #fff; border-width: 0 2px; } .arm-joint { position: absolute; top: 0; left: 0; --w: 5px; --h: 5px; transition: 0.3s; z-index: -1; } .arm-joint::after { position: absolute; border: solid var(--blue) 2px; background-color: #fff; --w: 10px; --h: 10px; top: calc(var(--m) * var(--h) * -0.25); left: calc(var(--m) * var(--w) * -0.25); } .arm::after { position: absolute; --w: 13px; --h: 7px; bottom: calc(var(--m) * -1px); left: calc(var(--m) * -3px); background-image: url(); } .arm.missed::after { background-image: url(); } .control { position: relative; --h: 60px; width: 100%; text-align: right; background-color: var(--dark-blue); } .cover { position: absolute; background-color: var(--machine-color); --top-size: calc(var(--m) * 20px); --bottom-size: calc(var(--m) * 10px); z-index: 4; } .top { top: 0; width: 100%; height: calc(var(--m) * 16px); } .collection-arrow { position: absolute; left: calc(var(--m) * 21px); top: calc(var(--m) * 9px); --w: 8px; --h: 4px; background-image: url(); filter: sepia(1) brightness(0.5); } .bottom { bottom: 0px; width: 100%; height: calc(var(--m) * 8px); background-color: var(--brown); } .left { bottom: 0px; left: 0px; height: calc(var(--m) * 170px); width: calc(var(--m) * 8px); } .right { top: 0; right: 0px; height: 100%; width: calc(var(--m) * 116px); } .instruction { position: absolute; --w: 51px; --h: 12px; background-image: url(); top: calc(var(--m) * 34px); right: calc(var(--m) * 11px); } .arm { position: absolute; --w: 7px; --h: 28px; background-color: #fff; box-shadow: 0 0 0 2px var(--blue); transition: 0.3s; margin-top: calc(var(--m) * -1px); margin-left: calc(var(--m) * -1px); } .claws { position: absolute; --w: 3px; --h: 10px; bottom: calc(var(--m) * -5px); left: calc(var(--m) * 2.5px); } .claws::before, .claws::after { top: 0; --w: 10px; --h: 16px; transition: 0.2s; } .claws::before { left: calc(var(--m) * -10px); background-image: url(); --close-angle: 45deg; transform-origin: top right; } .claws::after { left: calc(var(--m) * 2.5px); background-image: url(); --close-angle: -45deg; transform-origin: top left; } .arm.open .claws::before, .arm.open .claws::after { rotate: var(--close-angle); } .arm-joint::before { --w: 20px; --h: 16px; transform: scale(var(--scale)); margin-left: -15px; margin-top: var(--shadow-pos); background-image: url(); background-image: url(); z-index: -1; opacity: 0.5; } button { position: relative; z-index: 5; --w: 24px; --h: 24px; margin: 12px 12px 0 0; border: 0; background-color: transparent; background-image: url(); filter: sepia(1) brightness(0.4); pointer-events: none; cursor: pointer; } .hori-btn { transform: rotate(90deg); } .active { animation: pulse infinite 1s; pointer-events: all; } @keyframes pulse { 0%, 100% { filter: sepia(0); } 50% { filter: sepia(1); } } .toy.display { transform: scale(3); animation: forwards show-toy-1 0.8s; animation-delay: 1s; } .toy { position: absolute; transition: transform 1s, left 0.3s, top 0.3s; --m: 2; --w: 20px; --h: 27px; transform: rotate(var(--rotate-angle)); pointer-events: none; } .toy::after { content: ''; } .toy.bear::after { --w: 24px; --h: 30px; top: calc(var(--m) * -2px); left: calc(var(--m) * -2px); background-image: url(); } .toy.bear.grabbed::after { background-image: url(); } .toy-wrapper .toy.bear::after { background-image: url(); } .toy.bunny { --w: 20px; --h: 29px; } .toy.bunny::after { --w: 24px; --h: 32px; top: calc(var(--m) * -2px); left: calc(var(--m) * -2px); background-image: url(); } .toy.bunny.grabbed::after { background-image: url(); } .toy-wrapper .toy.bunny::after { background-image: url(); } .toy.golem::after { --w: 26px; --h: 30px; top: calc(var(--m) * -1px); left: calc(var(--m) * -3px); background-image: url(); } .toy.golem.grabbed::after { background-image: url(); } .toy-wrapper .toy.golem::after { background-image: url(); } .toy.cucumber { --w: 16px; --h: 28px; } .toy.cucumber::after { --w: 16px; --h: 30px; top: calc(var(--m) * 1px); left: 0; background-image: url(); } .toy.cucumber.grabbed::after { background-image: url(); } .toy-wrapper .toy.cucumber::after { background-image: url(); } .toy.penguin { --w: 24px; --h: 22px; } .toy.penguin::after { --w: 26px; --h: 24px; top: calc(var(--m) * -1px); left: calc(var(--m) * -1px); background-image: url(); } .toy.penguin.grabbed::after { background-image: url(); } .toy-wrapper .toy.penguin::after { background-image: url(); } .toy.robot { --w: 20px; --h: 30px; } .toy.robot::after { --w: 24px; --h: 32px; top: calc(var(--m) * -1px); left: calc(var(--m) * -2px); background-image: url(); } .toy.robot.grabbed::after { background-image: url(); } .toy-wrapper .toy.robot::after { background-image: url(); } .toy.selected { pointer-events: all; } @keyframes show-toy-1 { 0% { opacity: 1; transform: scale(3) translateY(0); } 30% { opacity: 0; } 100% { opacity: 0; transform: scale(1) translateY(-100vh); } } .control .collection-point { filter: brightness(0.8); bottom: calc(var(--m) * 8px); } .sign { position: fixed; color: var(--brown); bottom: 10px; right: 10px; font-size: 10px; } a { color: var(--brown); text-decoration: none; } a:hover { text-decoration: underline; }
JS
Copy
const elements = { clawMachine: document.querySelector('.claw-machine'), box: document.querySelector('.box'), collectionBox: document.querySelector('.collection-box'), collectionArrow: document.querySelector('.collection-arrow'), toys: [], } const settings = { targetToy: null, collectedNumber: 0, } const m = 2 const toys = { bear: { w: 20 * m, h: 27 * m, }, bunny: { w: 20 * m, h: 29 * m, }, golem: { w: 20 * m, h: 27 * m, }, cucumber: { w: 16 * m, h: 28 * m, }, penguin: { w: 24 * m, h: 22 * m, }, robot: { w: 20 * m, h: 30 * m, }, } const sortedToys = [...Object.keys(toys), ...Object.keys(toys)].sort( () => 0.5 - Math.random(), ) const cornerBuffer = 16 const machineBuffer = { x: 36, y: 16, } const radToDeg = rad => Math.round(rad * (180 / Math.PI)) const calcX = (i, n) => i % n const calcY = (i, n) => Math.floor(i / n) const { width: machineWidth, height: machineHeight, top: machineTop, } = document.querySelector('.claw-machine').getBoundingClientRect() const { height: machineTopHeight } = document .querySelector('.machine-top') .getBoundingClientRect() const { height: machineBottomHeight, top: machineBottomTop } = document .querySelector('.machine-bottom') .getBoundingClientRect() const maxArmLength = machineBottomTop - machineTop - machineBuffer.y const adjustAngle = angle => { const adjustedAngle = angle % 360 return adjustedAngle < 0 ? adjustedAngle + 360 : adjustedAngle } const randomN = (min, max) => { return Math.round(min - 0.5 + Math.random() * (max - min + 1)) } //* classes *// class Button { constructor({ className, action, isLocked, pressAction, releaseAction }) { Object.assign(this, { el: document.querySelector(`.${className}`), isLocked, }) this.el.addEventListener('click', action) ;['mousedown', 'touchstart'].forEach(action => this.el.addEventListener(action, pressAction), ) ;['mouseup', 'touchend'].forEach(action => this.el.addEventListener(action, releaseAction), ) if (!isLocked) this.activate() } activate() { this.isLocked = false this.el.classList.add('active') } deactivate() { this.isLocked = true this.el.classList.remove('active') } } class WorldObject { constructor(props) { Object.assign(this, { x: 0, y: 0, z: 0, angle: 0, transformOrigin: { x: 0, y: 0 }, interval: null, default: {}, moveWith: [], el: props.className && document.querySelector(`.${props.className}`), ...props, }) this.setStyles() if (props.className) { const { width, height } = this.el.getBoundingClientRect() this.w = width this.h = height } ;['x', 'y', 'w', 'h'].forEach(key => { this.default[key] = this[key] }) } setStyles() { Object.assign(this.el.style, { left: `${this.x}px`, top: !this.bottom && `${this.y}px`, bottom: this.bottom, width: `${this.w}px`, height: `${this.h}px`, transformOrigin: this.transformOrigin, }) this.el.style.zIndex = this.z } setClawPos(clawPos) { this.clawPos = clawPos } setTransformOrigin(transformOrigin) { this.transformOrigin = transformOrigin === 'center' ? 'center' : `${transformOrigin.x}px ${transformOrigin.y}px` this.setStyles() } handleNext(next) { clearInterval(this.interval) if (next) next() } resumeMove({ moveKey, target, moveTime, next }) { this.interval = null this.move({ moveKey, target, moveTime, next }) } resizeShadow() { elements.box.style.setProperty('--scale', 0.5 + this.h / maxArmLength / 2) } move({ moveKey, target, moveTime, next }) { if (this.interval) { this.handleNext(next) } else { const moveTarget = target || this.default[moveKey] this.interval = setInterval(() => { const distance = Math.abs(this[moveKey] - moveTarget) < 10 ? Math.abs(this[moveKey] - moveTarget) : 10 const increment = this[moveKey] > moveTarget ? -distance : distance if ( increment > 0 ? this[moveKey] < moveTarget : this[moveKey] > moveTarget ) { this[moveKey] += increment this.setStyles() if (moveKey === 'h') this.resizeShadow() if (this.moveWith.length) { this.moveWith.forEach(obj => { if (!obj) return obj[moveKey === 'h' ? 'y' : moveKey] += increment obj.setStyles() }) } } else { this.handleNext(next) } }, moveTime || 100) } } distanceBetween(target) { return Math.round( Math.sqrt( Math.pow(this.x - target.x, 2) + Math.pow(this.y - target.y, 2), ), ) } } class Toy extends WorldObject { constructor(props) { const toyType = sortedToys[props.index] const size = toys[toyType] super({ el: Object.assign(document.createElement('div'), { className: `toy pix ${toyType}`, }), x: cornerBuffer + calcX(props.index, 4) * ((machineWidth - cornerBuffer * 3) / 4) + size.w / 2 + randomN(-6, 6), y: machineBottomTop - machineTop + cornerBuffer + calcY(props.index, 4) * ((machineBottomHeight - cornerBuffer * 2) / 3) - size.h / 2 + randomN(-2, 2), z: 0, toyType, ...size, ...props, }) elements.box.append(this.el) const toy = this this.el.addEventListener('click', () => this.collectToy(toy)) elements.toys.push(this) } collectToy(toy) { toy.el.classList.remove('selected') toy.x = machineWidth / 2 - toy.w / 2 toy.y = machineHeight / 2 - toy.h / 2 toy.z = 7 toy.el.style.setProperty('--rotate-angle', '0deg') toy.setTransformOrigin('center') toy.el.classList.add('display') elements.clawMachine.classList.add('show-overlay') settings.collectedNumber++ elements.collectionBox.appendChild( Object.assign(document.createElement('div'), { className: `toy-wrapper ${ settings.collectedNumber > 6 ? 'squeeze-in' : '' }`, innerHTML: `<div class="toy pix ${toy.toyType}"></div>`, }), ) setTimeout(() => { elements.clawMachine.classList.remove('show-overlay') if (!document.querySelector('.selected')) elements.collectionArrow.classList.remove('active') }, 1000) } setRotateAngle() { const angle = radToDeg( Math.atan2( this.y + this.h / 2 - this.clawPos.y, this.x + this.w / 2 - this.clawPos.x, ), ) - 90 const adjustedAngle = Math.round(adjustAngle(angle)) this.angle = adjustedAngle < 180 ? adjustedAngle * -1 : 360 - adjustedAngle this.el.style.setProperty('--rotate-angle', `${this.angle}deg`) } } //* set up *// elements.box.style.setProperty('--shadow-pos', `${maxArmLength}px`) const armJoint = new WorldObject({ className: 'arm-joint', }) const vertRail = new WorldObject({ className: 'vert', moveWith: [null, armJoint], }) const arm = new WorldObject({ className: 'arm', }) armJoint.resizeShadow() armJoint.move({ moveKey: 'y', target: machineTopHeight - machineBuffer.y, moveTime: 50, next: () => vertRail.resumeMove({ moveKey: 'x', target: machineBuffer.x, moveTime: 50, next: () => { Object.assign(armJoint.default, { y: machineTopHeight - machineBuffer.y, x: machineBuffer.x, }) Object.assign(vertRail.default, { x: machineBuffer.x, }) activateHoriBtn() }, }), }) const doOverlap = (a, b) => { return b.x > a.x && b.x < a.x + a.w && b.y > a.y && b.y < a.y + a.h } const getClosestToy = () => { const claw = { y: armJoint.y + maxArmLength + machineBuffer.y + 7, x: armJoint.x + 7, w: 40, h: 32, } const overlappedToys = elements.toys.filter(t => { return doOverlap(t, claw) }) if (overlappedToys.length) { const toy = overlappedToys.sort((a, b) => b.index - a.index)[0] toy.setTransformOrigin({ x: claw.x - toy.x, y: claw.y - toy.y, }) toy.setClawPos({ x: claw.x, y: claw.y, }) settings.targetToy = toy } } new Array(12).fill('').forEach((_, i) => { if (i === 8) return new Toy({ index: i }) }) const stopHoriBtnAndActivateVertBtn = () => { armJoint.interval = null horiBtn.deactivate() vertBtn.activate() } const activateHoriBtn = () => { horiBtn.activate() ;[vertRail, armJoint, arm].forEach(c => (c.interval = null)) } const dropToy = () => { arm.el.classList.add('open') if (settings.targetToy) { settings.targetToy.z = 3 settings.targetToy.move({ moveKey: 'y', target: machineHeight - settings.targetToy.h - 30, moveTime: 50, }) ;[vertRail, armJoint, arm].forEach(obj => (obj.moveWith[0] = null)) } setTimeout(() => { arm.el.classList.remove('open') activateHoriBtn() if (settings.targetToy) { settings.targetToy.el.classList.add('selected') elements.collectionArrow.classList.add('active') settings.targetToy = null } }, 700) } const grabToy = () => { if (settings.targetToy) { ;[vertRail, armJoint, arm].forEach( obj => (obj.moveWith[0] = settings.targetToy), ) settings.targetToy.setRotateAngle() settings.targetToy.el.classList.add('grabbed') } else { arm.el.classList.add('missed') } } const horiBtn = new Button({ className: 'hori-btn', isLocked: true, pressAction: () => { arm.el.classList.remove('missed') vertRail.move({ moveKey: 'x', target: machineWidth - armJoint.w - machineBuffer.x, next: stopHoriBtnAndActivateVertBtn, }) }, releaseAction: () => { clearInterval(vertRail.interval) stopHoriBtnAndActivateVertBtn() }, }) const vertBtn = new Button({ className: 'vert-btn', isLocked: true, pressAction: () => { if (vertBtn.isLocked) return armJoint.move({ moveKey: 'y', target: machineBuffer.y, }) }, releaseAction: () => { clearInterval(armJoint.interval) vertBtn.deactivate() getClosestToy() setTimeout(() => { arm.el.classList.add('open') arm.move({ moveKey: 'h', target: maxArmLength, next: () => setTimeout(() => { arm.el.classList.remove('open') grabToy() arm.resumeMove({ moveKey: 'h', next: () => { vertRail.resumeMove({ moveKey: 'x', next: () => { armJoint.resumeMove({ moveKey: 'y', next: dropToy, }) }, }) }, }) }, 500), }) }, 500) }, })