<!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>