let canvas = document.getElementById('canvas'), ctx = canvas.getContext('2d'), scoreBlock = document.getElementById('score'), scoreCount = 0, bestScoreBlock = document.getElementById('best-score'), dir = '', // snake direction diff = 'Easy', // difficulty diffBlock = document.getElementById('difficulty'), btnChange = document.getElementById('changeDif'); const config = { // General setting sizeCell: 24, sizeFood: 24, step: 0, stepMax: 7, } const snake = { // Snake settings x: config.sizeCell, y: config.sizeCell, dirX: 0, // direction X dirY: 0, // direction Y body: [], maxBodySize: 1, } const snakeSkins = [ // array snake skins './img/snake/head.svg', ]; const snakeImages = [ imgHead = new Image(), ]; for ( let i = 0; i < snakeImages.length; i++ ) { snakeImages[i].src = snakeSkins[i]; } const food = { // food settings x: randomInt(0, canvas.width / config.sizeCell) * config.sizeCell, y: randomInt(0, canvas.height / config.sizeCell) * config.sizeCell, } const images = [ // array with images path './img/food/apple.svg', './img/food/carrot.svg', './img/food/eggplant.svg', './img/food/banana.svg', ]; let img = new Image(); img.src = images[0]; // the default will be the first image const bomb = { // bomb settings x: -config.sizeCell, // the default bomb will be hidden y: -config.sizeCell, }; const bombImg = new Image(); bombImg.src = './img/food/bomb.svg'; // аудио const audio = [ // array with audio './audio/eat.mp3', './audio/turn.mp3', './audio/dead.mp3', './audio/hit.mp3', ]; const audioNames = [ audioEat = new Audio(), audioTurn = new Audio(), audioDead = new Audio(), audioHit = new Audio(), ]; for ( let i = 0; i < audio.length; i++ ) { audioNames[i].src = audio[i]; } // canvas settings window.addEventListener('load', (e) => { if ( window.innerWidth <= 650 ) { canvas.width = 300; canvas.height = 300; ctx.fillStyle = '#000000' ctx.fillRect(0, 0, canvas.width, canvas.height); config.sizeCell = 15; config.sizeFood = 15; canvas.style = 'background-image: url("");'; restart(); } else if ( window.innerWidth > 650 ) { canvas.width = 600; canvas.height = 480; ctx.fillStyle = '#000000' ctx.fillRect(0, 0, canvas.width, canvas.height); config.sizeCell = 24; config.sizeFood = 24; canvas.style = 'background-image: url("");'; restart(); } }); // adaptive window.addEventListener('resize', (e) => { if ( window.innerWidth <= 650 ) { canvas.width = 300; canvas.height = 300; ctx.fillStyle = '#000000' ctx.fillRect(0, 0, canvas.width, canvas.height); config.sizeCell = 15; config.sizeFood = 15; canvas.style = 'background-image: url("");'; } else if ( window.innerWidth > 650 ) { canvas.width = 600; canvas.height = 480; ctx.fillStyle = '#000000' ctx.fillRect(0, 0, canvas.width, canvas.height); config.sizeCell = 24; config.sizeFood = 24; canvas.style = 'background-image: url("");'; } }); // score function score() { scoreCount++; bestScore(); if ( scoreCount > 15 ) config.stepMax = 5; else if ( scoreCount <= 15 ) config.stepMax = 6; drawScore(); } function drawScore() { scoreBlock.innerHTML = scoreCount; } function bestScore() { if ( !localStorage.getItem('best score') ) { localStorage.setItem('best score', 0); } if ( scoreCount > localStorage.getItem('best score') ) { localStorage.setItem('best score', scoreCount); } bestScoreBlock.innerHTML = localStorage.getItem('best score'); } // game function gameLoop() { requestAnimationFrame(gameLoop); if ( ++config.step < config.stepMax ) return; config.step = 0; ctx.clearRect(0, 0, canvas.width, canvas.height); // очистка canvas drawFood(); drawSnake(); if (diff === 'Hard') { ctx.strokeStyle = '#f00'; ctx.lineWidth = 5; ctx.strokeRect(0, 0, canvas.width, canvas.height); drawBomb(); }; } gameLoop(); // difficulty, restart function checkBorder() { // A function that checks for going beyond the border // x if ( snake.x < 0 ) { snake.x = canvas.width - config.sizeCell; } else if ( snake.x >= canvas.width ) { snake.x = 0; } // y if ( snake.y < 0 ) { snake.y = canvas.height - config.sizeCell; } else if ( snake.y >= canvas.height ) { snake.y = 0; } } function withoutBorder() { // A function that checks if the snake is touched border if ( snake.x < 0 ) { audioPlay('dead'); restart(); } else if ( snake.x >= canvas.width ) { audioPlay('dead'); restart(); } if ( snake.y < 0 ) { audioPlay('dead'); restart(); } else if ( snake.y >= canvas.height ) { audioPlay('dead'); restart(); } } function restart() { // A function restart game config.stepMax = 6; scoreCount = 0; drawScore(); snake.x = config.sizeCell; snake.y = config.sizeCell; snake.body = []; snake.maxBodySize = 1; snake.dirX = 0; snake.dirY = 0; dir = ''; randomPosFood(); if ( diff === 'Hard' ) randomPosBomb(); } // draw snake function drawSnake() { snake.x += snake.dirX; snake.y += snake.dirY; if (diff === 'Easy') checkBorder(); if (diff === 'Hard') withoutBorder(); // work with snake length snake.body.unshift({x: snake.x, y: snake.y}); if ( snake.body.length > snake.maxBodySize ) { snake.body.pop(); } snake.body.forEach((e, index) => { snakeStyles(e, index); // snake styles if ( e.x == food.x && e.y == food.y ) { // If snake ate food audioPlay('eat'); score(); randomPosFood(); snake.maxBodySize++; if ( diff === 'Hard' ) { randomPosBomb(); } } if ( diff === 'Hard' ) { // if snake has touched the bomd, reduce the length of the snake if ( e.x === bomb.x && e.y === bomb.y ) { if ( scoreCount >= 2 ) { audioPlay('hit'); scoreCount = Math.ceil(scoreCount / 2); snake.maxBodySize = scoreCount + 1; for ( let i = 0; i < snake.maxBodySize; i++ ) { snake.body.pop(); } drawScore(); randomPosFood(); randomPosBomb(); } else { audioPlay('dead'); restart() } } } // Checking if the snake has touched the tail for ( let i = index + 1; i < snake.body.length; i++ ) { if ( e.x === snake.body[i].x && e.y === snake.body[i].y ) { audioPlay('dead'); restart(); } } }); } // snake styles function snakeStyles(e, index) { if ( index === 0 ) { // for the first snake element(head) ctx.drawImage(snakeImages[0], e.x, e.y, config.sizeCell, config.sizeCell); } else { // other elements ctx.fillStyle = '#093D14'; ctx.strokeStyle = '#071510'; ctx.lineWidth = 1; ctx.fillRect(e.x, e.y, config.sizeCell, config.sizeCell); ctx.strokeRect(e.x, e.y, config.sizeCell - 1, config.sizeCell - 1); } } // draw food function drawFood() { ctx.drawImage(img, food.x, food.y, config.sizeFood, config.sizeFood); } // draw bomb function drawBomb() { // функция добовления бомбы ctx.drawImage(bombImg, bomb.x, bomb.y, config.sizeFood, config.sizeFood); } // Load the best score document.addEventListener('load', bestScore()); // constrols function turnUp() { if ( dir != 'down' ) { audioPlay('turn'); dir = 'up'; snake.dirY = -config.sizeCell; snake.dirX = 0; } } function turnLeft() { if ( dir != 'right' ) { audioPlay('turn'); dir = 'left'; snake.dirX = -config.sizeCell; snake.dirY = 0; } } function turnDown() { if ( dir != 'up' ) { audioPlay('turn'); dir = 'down'; snake.dirY = config.sizeCell; snake.dirX = 0; } } function turnRight() { if ( dir != 'left' ) { audioPlay('turn'); dir = 'right'; snake.dirX = config.sizeCell; snake.dirY = 0; } } document.addEventListener('keydown', (e) => { if ( e.keyCode == 87 || e.keyCode == 38 ) { // W (up) or arrow up turnUp(); } else if ( e.keyCode == 65 || e.keyCode == 37 ) { // A (left) or arrow left turnLeft(); } else if ( e.keyCode == 83 || e.keyCode == 40 ) { // S (down) or arrow down turnDown(); } else if ( e.keyCode == 68 || e.keyCode == 39 ) { // D (right) or arrow right turnRight(); } }); // touch const isMobile = { Android: function() { return navigator.userAgent.match(/Android/i); }, BlackBerry: function() { return navigator.userAgent.match(/BlackBerry/i); }, iOS: function() { return navigator.userAgent.match(/iPhone|iPad|iPod/i); }, Opera: function() { return navigator.userAgent.match(/Opera Mini/i); }, Windows: function() { return navigator.userAgent.match(/IEMobile/i); }, any: function() { return (isMobile.Android() || isMobile.BlackBerry() || isMobile.iOS() || isMobile.Opera() || isMobile.Windows()); } }; if ( isMobile.any() ) { alert('Please click any where for play game sounds'); document.addEventListener('touchstart', handleTouchStart); document.addEventListener('touchmove', handleTouchMove); let x1 = null; let y1 = null; function handleTouchStart(event) { const firstTouch = event.touches[0]; x1 = firstTouch.clientX; y1 = firstTouch.clientY; } function handleTouchMove(event) { if ( !x1 || !y1 ) return false; // check swipe let x2 = event.touches[0].clientX; let y2 = event.touches[0].clientY; let xDiff = x2 - x1; let yDiff = y2 - y1; if ( Math.abs(xDiff) > Math.abs(yDiff) ) { // swipe left or right xDiff > 0 ? turnRight() : turnLeft(); } else { // swipe up or down yDiff < 0 ? turnUp() : turnDown(); } x1 = null; y1 = null; } } // change difficulty btnChange.addEventListener('click', (e) => { if ( diff === 'Easy' ) { diff = 'Hard'; diffBlock.innerHTML = 'Hard'; restart(); } else { diff = 'Easy'; diffBlock.innerHTML = 'Easy'; restart(); } }); // Additional functions function randomInt(min, max) { return Math.floor(Math.random() * (max - min) + min); } function randomImg() { // random img ( for food ) let imgCount = randomInt(0, images.length); let imgPath = images[imgCount]; img.src = imgPath; return img; } function randomPosFood() { // random food position ctx.drawImage(randomImg(), food.x, food.y, config.sizeFood, config.sizeFood); food.x = randomInt(0, canvas.width / config.sizeCell) * config.sizeCell; food.y = randomInt(0, canvas.height / config.sizeCell) * config.sizeCell; drawFood(); } function randomPosBomb() { // random bomb position let chance = randomInt(1, 5); if ( chance === 3 ) { bomb.x = randomInt(0, canvas.width / config.sizeCell) * config.sizeCell; bomb.y = randomInt(0, canvas.height / config.sizeCell) * config.sizeCell; drawBomb(); } else { bomb.x = -config.sizeCell; bomb.y = -config.sizeCell; drawBomb(); } } function audioPlay(name) { // play audio if ( name === 'eat' ) { audioNames[0].play(); } if ( name === 'turn' ) { audioNames[1].play(); } if ( name === 'dead' ) { audioNames[2].play(); } if ( name === 'hit' ) { audioNames[3].play(); } }