Когда тебе тяжело, всегда напоминай себе о том, что если ты сдашься, лучше не станет.
5-декабря-2023, 21:36 6 0
Как украсить сайт к новому году? Елки, снеговики, гирлянды, снежинки, а может лучше игру!? Хочу Вам предложить рождественскую игру для вашего веб сайта Помощник Санты. Игра простая , кликаем удерживаем определённое время левую кнопку мыши, в это время начинает расти вертикальная палочка в виде рождественской конфеты. По сути это мост, наша задача помочь Санте переместиться на другую колонну. Вот и всё. Таким способом можно развлечь ваших посетителей!
HTML
<a href="https://g.hmz.ie/santa" target="_blank"><img style="position: absolute; top: 0; right: 0; border: 0; z-index:1000" src="https://camo.githubusercontent.com/38ef81f8aca64bb9a64448d0d70f1308ef5341ab/68747470733a2f2f73332e616d617a6f6e6177732e636f6d2f6769746875622f726962626f6e732f666f726b6d655f72696768745f6461726b626c75655f3132313632312e706e67" alt="Fork me on GitHub" data-canonical-src="https://s3.amazonaws.com/github/ribbons/forkme_right_darkblue_121621.png"></a>
<!-- Based on Original work by Hunor Marton
https://codepen.io/HunorMarton/details/xxOMQKg
-->
CSS
html,
body {
height: 100%;
margin: 0;
}
body {
font-family: Arial, Verdana, sans-serif;
cursor: grab;
display: flex;
justify-content: center;
align-items: center;
color: white;
background-color: #1e1a33;
}
div,
i {
user-select: none;
pointer-events: none;
}
i {
position: fixed;
color: white;
top: -10%;
z-index: 9999;
animation-name: snowflakes-fall, snowflakes-shake;
animation-duration: 10s, 3s;
animation-timing-function: linear, ease-in-out;
animation-iteration-count: infinite, infinite;
animation-play-state: running, running;
}
@keyframes snowflakes-fall {
0% {
top: -10%;
}
100% {
top: 100%;
}
}
@keyframes snowflakes-shake {
0% {
transform: translateX(0px);
}
50% {
transform: translateX(80px);
}
100% {
transform: translateX(0px);
}
}
JS
let status = "waiting";
let lastTimestamp;
let santaX;
let santaY;
let sceneOffset;
let score = 0;
let platforms = [];
let sticks = [];
let trees = [];
let clouds = [];
const config = {
canvasWidth: 375,
canvasHeight: 375,
platformHeight: 100,
santaDistanceFromEdge: 10,
paddingX: 100,
perfectAreaSize: 10,
backgroundSpeedMultiplier: 0.2,
speed: 4,
santaWidth: 17,
santaHeight: 30
};
const colours = {
lightBg: "#62AFB9",
medBg: "#182757",
darkBg: "#0D5B66",
lightHill: "#E9E9E9",
medHill: "#34A65F",
darkHill: "#07133A",
platform: "#9B4546",
platformTop: "#620E0E",
em: "#CC231E",
skin: "#CF6D60"
};
const hills = [
{
baseHeight: 120,
amplitude: 20,
stretch: 0.5,
colour: colours.lightHill
},
{
baseHeight: 100,
amplitude: 10,
stretch: 1,
colour: colours.medHill
},
{
baseHeight: 70,
amplitude: 20,
stretch: 0.5,
colour: colours.darkHill
}
];
const scoreElement = createElementStyle(
"div",
`position:absolute;top:1.5em;font-size:5em;font-weight:900;text-shadow:${addShadow(
colours.darkHill,
7
)}`
);
const canvas = createElementStyle("canvas");
const introductionElement = createElementStyle(
"div",
`font-size:1.2em;position:absolute;text-align:center;transition:opacity 2s;width:250px`,
"Press and hold anywhere to stretch out a sugar cane, it has to be the exact length or Santa will fall down"
);
const perfectElement = createElementStyle(
"div",
"position:absolute;opacity:0;transition:opacity 2s",
"Double Score"
);
const restartButton = createElementStyle(
"button",
`width:120px;height:120px;position:absolute;border-radius:50%;color:white;background-color:${colours.em};border:none;font-weight:700;font-size:1.2em;display:none;cursor:pointer`,
"RESTART"
);
// Adding snow
// for (let i = 0; i <= 30; i++) {
// createElementStyle(
// "i",
// `font-size: ${3 * Math.random()}em;left: ${
// 100 * Math.random()
// }%; animation-delay: ${10 * Math.random()}s, ${2 * Math.random()}s`,
// "."
// );
// }
canvas.width = window.innerWidth;
canvas.height = window.innerHeight;
const ctx = canvas.getContext("2d");
Array.prototype.last = function () {
return this[this.length - 1];
};
Math.sinus = function (degree) {
return Math.sin((degree / 180) * Math.PI);
};
window.addEventListener("keydown", function (event) {
if (event.key == " ") {
event.preventDefault();
resetGame();
return;
}
});
["mousedown", "touchstart"].forEach(function (evt) {
window.addEventListener(evt, function (event) {
if (status == "waiting") {
lastTimestamp = undefined;
introductionElement.style.opacity = 0;
status = "stretching";
window.requestAnimationFrame(animate);
}
});
});
["mouseup", "touchend"].forEach(function (evt) {
window.addEventListener(evt, function (event) {
if (status == "stretching") {
status = "turning";
}
});
});
window.addEventListener("resize", function (event) {
canvas.width = window.innerWidth;
canvas.height = window.innerHeight;
draw();
});
restartButton.addEventListener("click", function (event) {
event.preventDefault();
resetGame();
restartButton.style.display = "none";
});
window.requestAnimationFrame(animate);
resetGame();
function resetGame() {
status = "waiting";
lastTimestamp = undefined;
sceneOffset = 0;
score = 0;
introductionElement.style.opacity = 1;
perfectElement.style.opacity = 0;
restartButton.style.display = "none";
scoreElement.innerText = score;
platforms = [{ x: 50, w: 50 }];
santaX = platforms[0].x + platforms[0].w - config.santaDistanceFromEdge;
santaY = 0;
sticks = [{ x: platforms[0].x + platforms[0].w, length: 0, rotation: 0 }];
trees = [];
clouds = [];
for (let i = 0; i <= 20; i++) {
if (i <= 3) generatePlatform();
generateTree();
generateCloud();
}
draw();
}
function generateCloud() {
const minimumGap = 60;
const maximumGap = 300;
const lastCloud = clouds[clouds.length - 1];
let furthestX = lastCloud ? lastCloud.x : 0;
const x =
furthestX +
minimumGap +
Math.floor(Math.random() * (maximumGap - minimumGap));
const y =
minimumGap +
Math.floor(Math.random() * (maximumGap - minimumGap)) -
window.innerHeight / 1.2;
const w = Math.floor(Math.random() * 15 + 15);
clouds.push({ x, y, w });
}
function generateTree() {
const minimumGap = 30;
const maximumGap = 150;
const lastTree = trees[trees.length - 1];
let furthestX = lastTree ? lastTree.x : 0;
const x =
furthestX +
minimumGap +
Math.floor(Math.random() * (maximumGap - minimumGap));
const treeColors = [colours.lightHill, colours.medBg, colours.medHill];
const color = treeColors[Math.floor(Math.random() * 3)];
trees.push({ x, color });
}
function generatePlatform() {
const minimumGap = 40;
const maximumGap = 200;
const minimumWidth = 20;
const maximumWidth = 100;
const lastPlatform = platforms[platforms.length - 1];
let furthestX = lastPlatform.x + lastPlatform.w;
const x =
furthestX +
minimumGap +
Math.floor(Math.random() * (maximumGap - minimumGap));
const w =
minimumWidth + Math.floor(Math.random() * (maximumWidth - minimumWidth));
platforms.push({ x, w });
}
function animate(timestamp) {
if (!lastTimestamp) {
lastTimestamp = timestamp;
window.requestAnimationFrame(animate);
return;
}
switch (status) {
case "waiting":
return;
case "stretching": {
sticks.last().length += (timestamp - lastTimestamp) / config.speed;
break;
}
case "turning": {
sticks.last().rotation += (timestamp - lastTimestamp) / config.speed;
if (sticks.last().rotation > 90) {
sticks.last().rotation = 90;
const [nextPlatform, perfectHit] = thePlatformTheStickHits();
if (nextPlatform) {
score += perfectHit ? 2 : 1;
scoreElement.innerText = score;
if (perfectHit) {
perfectElement.style.opacity = 1;
setTimeout(() => (perfectElement.style.opacity = 0), 1000);
}
generatePlatform();
generateTree();
generateTree();
generateCloud();
generateCloud();
}
status = "walking";
}
break;
}
case "walking": {
santaX += (timestamp - lastTimestamp) / config.speed;
const [nextPlatform] = thePlatformTheStickHits();
if (nextPlatform) {
const maxSantaX =
nextPlatform.x + nextPlatform.w - config.santaDistanceFromEdge;
if (santaX > maxSantaX) {
santaX = maxSantaX;
status = "transitioning";
}
} else {
const maxSantaX =
sticks.last().x + sticks.last().length + config.santaWidth;
if (santaX > maxSantaX) {
santaX = maxSantaX;
status = "falling";
}
}
break;
}
case "transitioning": {
sceneOffset += (timestamp - lastTimestamp) / (config.speed / 2);
const [nextPlatform] = thePlatformTheStickHits();
if (sceneOffset > nextPlatform.x + nextPlatform.w - config.paddingX) {
sticks.push({
x: nextPlatform.x + nextPlatform.w,
length: 0,
rotation: 0
});
status = "waiting";
}
break;
}
case "falling": {
if (sticks.last().rotation < 180)
sticks.last().rotation += (timestamp - lastTimestamp) / config.speed;
santaY += (timestamp - lastTimestamp) / (config.speed / 2);
const maxSantaY =
config.platformHeight +
100 +
(window.innerHeight - config.canvasHeight) / 2;
if (santaY > maxSantaY) {
restartButton.style.display = "block";
return;
}
break;
}
default:
throw Error("Wrong status");
}
draw();
window.requestAnimationFrame(animate);
lastTimestamp = timestamp;
}
function thePlatformTheStickHits() {
if (sticks.last().rotation != 90)
throw Error(`Stick is ${sticks.last().rotation}°`);
const stickFarX = sticks.last().x + sticks.last().length;
const platformTheStickHits = platforms.find(
(platform) => platform.x < stickFarX && stickFarX < platform.x + platform.w
);
if (
platformTheStickHits &&
platformTheStickHits.x +
platformTheStickHits.w / 2 -
config.perfectAreaSize / 2 <
stickFarX &&
stickFarX <
platformTheStickHits.x +
platformTheStickHits.w / 2 +
config.perfectAreaSize / 2
)
return [platformTheStickHits, true];
return [platformTheStickHits, false];
}
function draw() {
ctx.save();
ctx.clearRect(0, 0, window.innerWidth, window.innerHeight);
drawBackground();
ctx.translate(
(window.innerWidth - config.canvasWidth) / 2 - sceneOffset,
(window.innerHeight - config.canvasHeight) / 2
);
drawPlatforms();
drawSanta();
drawSticks();
ctx.restore();
}
function drawPlatforms() {
platforms.forEach(({ x, w }) => {
let newX = x + 3;
let newW = w - 6;
let platformHeight =
config.platformHeight + (window.innerHeight - config.canvasHeight) / 2;
ctx.fillStyle = colours.platform;
ctx.fillRect(
newX,
config.canvasHeight - config.platformHeight,
newW,
platformHeight
);
for (let i = 1; i <= platformHeight / 10; ++i) {
let yGap = config.canvasHeight - config.platformHeight + i * 10;
ctx.moveTo(newX, yGap);
ctx.lineTo(newX + newW, yGap);
let xGap = i % 2 ? 0 : 10;
for (let j = 1; j < newW / 30; ++j) {
let x = j * 20 + xGap;
ctx.moveTo(newX + x, yGap);
ctx.lineTo(newX + x, yGap + 10);
}
ctx.strokeStyle = colours.platformTop;
ctx.stroke();
}
ctx.fillStyle = colours.platformTop;
ctx.fillRect(x, config.canvasHeight - config.platformHeight, w, 10);
if (sticks.last().x < x) {
ctx.fillStyle = "white";
ctx.fillRect(
x + w / 2 - config.perfectAreaSize / 2,
config.canvasHeight - config.platformHeight,
config.perfectAreaSize,
config.perfectAreaSize
);
}
});
}
function drawSanta() {
ctx.save();
ctx.fillStyle = "red";
ctx.translate(
santaX - config.santaWidth / 2,
santaY +
config.canvasHeight -
config.platformHeight -
config.santaHeight / 2
);
ctx.fillRect(
-config.santaWidth / 2,
-config.santaHeight / 2,
config.santaWidth,
config.santaHeight - 4
);
const legDistance = 5;
ctx.beginPath();
ctx.arc(legDistance, 11.5, 3, 0, Math.PI * 2, false);
ctx.fill();
ctx.beginPath();
ctx.arc(-legDistance, 11.5, 3, 0, Math.PI * 2, false);
ctx.fill();
ctx.beginPath();
ctx.fillStyle = colours.skin;
ctx.arc(5, -7, 3, 0, Math.PI * 2, false);
ctx.fill();
ctx.beginPath();
ctx.fillStyle = "white";
ctx.arc(7, -2, 3, 0, Math.PI * 2, false);
ctx.fill();
ctx.beginPath();
ctx.fillStyle = "red";
ctx.moveTo(-8, -13.5);
ctx.lineTo(-15, -3.5);
ctx.lineTo(-5, -7);
ctx.fill();
ctx.fillStyle = "white";
ctx.fillRect(-config.santaWidth / 2, -12, config.santaWidth, 3);
ctx.fillStyle = "black";
ctx.fillRect(-config.santaWidth / 2, 2, config.santaWidth, 2);
ctx.fillStyle = "white";
ctx.fillRect(-config.santaWidth / 2, 4, config.santaWidth, 4.5);
ctx.beginPath();
ctx.fillStyle = "white";
ctx.arc(-17, -2, 3, 0, Math.PI * 2, false);
ctx.fill();
ctx.restore();
}
function drawSticks() {
sticks.forEach((stick) => {
ctx.save();
ctx.translate(stick.x, config.canvasHeight - config.platformHeight);
ctx.rotate((Math.PI / 180) * stick.rotation);
ctx.beginPath();
ctx.lineWidth = 4;
ctx.moveTo(0, 0);
ctx.lineTo(0, -stick.length);
ctx.strokeStyle = ctx.createPattern(createCandyPattern(), "repeat");
ctx.stroke();
ctx.restore();
});
}
function drawBackground() {
var gradient = ctx.createRadialGradient(
window.innerWidth / 2,
window.innerHeight / 2,
0,
window.innerHeight / 2,
window.innerWidth / 2,
window.innerWidth
);
gradient.addColorStop(0, colours.lightBg);
gradient.addColorStop(1, colours.darkBg);
ctx.fillStyle = gradient;
ctx.fillRect(0, 0, window.innerWidth, window.innerHeight);
hills.forEach((hill) =>
drawHill(hill.baseHeight, hill.amplitude, hill.stretch, hill.colour)
);
trees.forEach((tree) => drawTree(tree.x, tree.color));
clouds.forEach((cloud) => drawCloud(cloud.x, cloud.y, cloud.w));
}
function drawHill(baseHeight, amplitude, stretch, color) {
ctx.beginPath();
ctx.moveTo(0, window.innerHeight);
ctx.lineTo(0, getHillY(0, baseHeight, amplitude, stretch));
for (let i = 0; i < window.innerWidth; i++) {
ctx.lineTo(i, getHillY(i, baseHeight, amplitude, stretch));
}
ctx.lineTo(window.innerWidth, window.innerHeight);
ctx.fillStyle = color;
ctx.fill();
}
function drawTree(x, color) {
ctx.save();
ctx.translate(
(-sceneOffset * config.backgroundSpeedMultiplier + x) * hills[1].stretch,
getTreeY(x, hills[1].baseHeight, hills[1].amplitude)
);
const treeTrunkHeight = 15;
const treeTrunkWidth = 10;
const treeCrownHeight = 60;
const treeCrownWidth = 30;
// Draw trunk
ctx.fillStyle = colours.darkHill;
ctx.fillRect(
-treeTrunkWidth / 2,
-treeTrunkHeight,
treeTrunkWidth,
treeTrunkHeight
);
// Draw crown
ctx.beginPath();
ctx.moveTo(-treeCrownWidth / 2, -treeTrunkHeight * 3);
ctx.lineTo(0, -(treeTrunkHeight + treeCrownHeight));
ctx.lineTo(treeCrownWidth / 2, -treeTrunkHeight * 3);
ctx.moveTo(-treeCrownWidth / 2, -treeTrunkHeight * 2);
ctx.lineTo(0, -(treeTrunkHeight / 2 + treeCrownHeight));
ctx.lineTo(treeCrownWidth / 2, -treeTrunkHeight * 2);
ctx.moveTo(-treeCrownWidth / 2, -treeTrunkHeight);
ctx.lineTo(0, -(treeTrunkHeight + treeCrownHeight / 2));
ctx.lineTo(treeCrownWidth / 2, -treeTrunkHeight);
ctx.fillStyle = color;
ctx.fill();
ctx.restore();
}
function drawCloud(x, y, width) {
ctx.save();
ctx.translate(
(-sceneOffset * config.backgroundSpeedMultiplier + x) * hills[1].stretch,
getTreeY(x, hills[1].baseHeight, hills[1].amplitude)
);
height = width * 1.5;
ctx.beginPath();
ctx.arc(x, y, width, Math.PI * 0.5, Math.PI * 1.5);
ctx.arc(x + height, y - width, height, Math.PI * 1, Math.PI * 2);
ctx.arc(x + height * 2, y - width, height, Math.PI * 1.2, Math.PI);
ctx.arc(x + width * 3, y, width, Math.PI * 1.5, Math.PI * 0.5);
ctx.moveTo(x + width * 3, y + width);
ctx.lineTo(x, y + width);
ctx.fillStyle = "rgba(255, 255, 255, .3)";
ctx.fill();
ctx.restore();
}
function createCandyPattern() {
const patternCanvas = document.createElement("canvas");
const pctx = patternCanvas.getContext("2d");
const max = 15;
let i = 0;
let x = 0;
let z = 90;
while (i < max) {
pctx.beginPath();
pctx.moveTo(0, x);
pctx.lineTo(0, z);
pctx.lineWidth = 24;
pctx.strokeStyle = "red";
pctx.stroke();
pctx.beginPath();
pctx.moveTo(0, x + 24);
pctx.lineTo(0, z + 24);
pctx.lineWidth = 24;
pctx.strokeStyle = "white";
pctx.stroke();
x += 48;
z += 48;
i++;
}
return patternCanvas;
}
function getHillY(windowX, baseHeight, amplitude, stretch) {
const sineBaseY = window.innerHeight - baseHeight;
return (
Math.sinus(
(sceneOffset * config.backgroundSpeedMultiplier + windowX) * stretch
) *
amplitude +
sineBaseY
);
}
function getTreeY(x, baseHeight, amplitude) {
const sineBaseY = window.innerHeight - baseHeight;
return Math.sinus(x) * amplitude + sineBaseY;
}
function createElementStyle(element, cssStyles = null, inner = null) {
const g = document.createElement(element);
if (cssStyles) g.style.cssText = cssStyles;
if (inner) g.innerHTML = inner;
document.body.appendChild(g);
return g;
}
function addShadow(colour, depth) {
let shadow = "";
for (let i = 0; i <= depth; i++) {
shadow += `${i}px ${i}px 0 ${colour}`;
shadow += i < depth ? ", " : "";
}
return shadow;
}
const funs = new Funs('light');
funs.signature();