vue3,贪吃蛇

2024-09-04 17:36

本文主要是介绍vue3,贪吃蛇,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

<template><div flex items-center justify-center h-100vh w-full relative><div class=""><div class="header"><div>得分: {{ score }}</div><div class="flex"><div w-100>速度: {{ speed }}</div><el-sliderv-model="speed":min="2":max="9":step="0.1"@change="changeSpeed"/></div></div><div class="flex" v-for="(i, idx) in array" :key="idx"><divclass="block flex items-center justify-center"v-for="(j, idx2) in i":key="idx"><divv-if="j"class="w-13 h-13":class="{ food: j == 2, snake: j == 1 }":style="{backgroundColor:j == 2? foodColor: idx === head[0] && idx2 === head[1]? snakeHeadColor: snakeColor,border: j == 2 ? '' : '1px solid #bdbdbd',borderRadius:j == 2? '12px': idx === head[0] && idx2 === head[1]? headRadius(): '4px',}"></div></div></div></div><divclass="absolute left-0 bottom-0 right-0 top-0 flex items-center justify-center pause-bg"v-if="isPause"><div text-center><div text-36 mb-20 class="text-#fff">游戏暂停</div><div text-14 class="text-#eee">按空格键开始/暂停</div></div></div><divclass="absolute left-0 bottom-0 right-0 top-0 flex items-center justify-center"v-if="isOver"><el-button type="success" size="large" @click="refresh">重新开始</el-button></div></div>
</template><script setup lang="ts">
import {ref,reactive,onMounted,toRefs,watch,computed,onUnmounted,
} from "vue";/*** @param { value } 0-空格子 1-蛇蛇 2-食物* @param { ROW } 行数* @param { COL } 列数* @param { direction } 蛇蛇移动方向 0-上 1-右 2-下 3-左*/
let ROW = 22;
let COL = 18;
let direction = 1;
let directionTemp = direction;
let food = ref([0, 0]);
let isEat = false;
let timer: any = null;
const isOver = ref(false);
const isPause = ref(false);
let score = 0;
const speed = ref(3.5);
const array = ref<any>([]);
const snake = ref<any>([]);// #67C23A #409EFF #E6A23C #F56C6C #909399 #ffa4e4
const snakeColor = ref("#67C23A");
const snakeHeadColor = ref("#3C9212");
const foodColor = ref("#F56C6C");
console.log("array", array);watch(() => snake.value,(v) => {// console.log("v", v);v.forEach((item) => {array.value[item[0]][item[1]] = 1;});},{ deep: true }
);
watch(() => food.value,(v) => {// console.log('v', v);array.value[v[0]][v[1]] = 2;}
);const init = () => {timer = setInterval(() => {if (isOver.value) {clearInterval(timer);return;}if (isPause.value) {return;}let next = [head.value[0], head.value[1]];direction = directionTemp;switch (direction) {case 0:next[0] -= 1;break;case 1:next[1] += 1;break;case 2:next[0] += 1;break;case 3:next[1] -= 1;break;}if (next[0] < 0 || next[0] >= ROW || next[1] < 0 || next[1] >= COL) {// 撞墙isOver.value = true;alert("game over");return;}if (array.value[next[0]][next[1]] == 1) {// 撞身体isOver.value = true;alert("game over");return;}if (array.value[next[0]][next[1]] == 2) {isEat = true;score++;if (score == 5) {speed.value += 0.4;} else if (score == 10) {speed.value += 0.4;} else if (score == 15) {speed.value += 0.4;} else if (score == 20) {speed.value += 0.5;} else if (score == 25) {speed.value += 0.5;} else if (score == 30) {speed.value += 0.5;} else if (score == 35) {speed.value += 0.6;} else if (score == 40) {speed.value += 0.6;}// 最多保留一位小数speed.value = Math.round(speed.value * 10) / 10;}/*** 移动 && 检测是否进食*/snake.value.unshift(next);if (!isEat) {array.value[snake.value[snake.value.length - 1][0]][snake.value[snake.value.length - 1][1]] = 0;snake.value.pop();} else {isEat = false;newFoodPosition();}array.value[next[0]][next[1]] = 1;}, 1000 / speed.value);
};
const newFoodPosition = () => {// 生成新的食物点,并且不能在蛇的身体let x = Math.floor(Math.random() * ROW);let y = Math.floor(Math.random() * COL);while (snake.value.some((item) => item[0] == x && item[1] == y)) {x = Math.floor(Math.random() * ROW);y = Math.floor(Math.random() * COL);}food.value = [x, y];
};
const head = computed(() => {return snake.value[0];
});const keydown = (e: { keyCode: number; preventDefault: any }) => {e.preventDefault();if (e.keyCode == 37) {//左if (direction == 1) return;directionTemp = 3;} else if (e.keyCode == 38) {//上if (direction == 2) return;directionTemp = 0;} else if (e.keyCode == 39) {//右if (direction == 3) return;directionTemp = 1;} else if (e.keyCode == 40) {//下if (direction == 0) return;directionTemp = 2;} else if (e.keyCode == 32) {// 空格键暂停isPause.value = !isPause.value;}
};const changeSpeed = (v) => {};
const headRadius = () => {switch (direction) {case 0:return "12px 12px 0px 0px";case 1:return "0px 12px 12px 0px";case 2:return "0px 0px 12px 12px";case 3:return "12px 0px 0px 12px";}
};
watch(() => speed.value,(v) => {clearInterval(timer);init();}
);
const start = () => {clearInterval(timer);array.value = Array.from({ length: ROW }, () =>Array.from({ length: COL }, () => 0));snake.value = [[2, 7],[2, 6],[2, 5],[2, 4],[2, 3],];newFoodPosition();init();
};
const refresh = () =>{window.location.reload();
}
onMounted(() => {// console.log("game start");start();document.addEventListener("keydown", keydown);
});
onUnmounted(() => {clearInterval(timer);document.removeEventListener("keydown", keydown);
});
</script><style lang="scss" scoped>
$width: 14px;
.block {width: $width;height: $width;background-color: #e4e4e4;margin: 1px;
}
.food {clip-path: polygon(10% 10%, 90% 10%, 90% 90%, 10% 90%);animation: food alternate 1.5s ease-in-out infinite;
}
@keyframes food {0% {transform: scale(1);}25% {transform: scale(0.92);filter: brightness(0.9);}50% {transform: scale(1.1);}75% {transform: scale(1.24);filter: brightness(1.15);}100% {transform: scale(1.16);}
}
.snake {background-image: linear-gradient(45deg,rgba(255, 255, 255, 0.16) 25%,transparent 25%,transparent 50%,rgba(255, 255, 255, 0.16) 50%,rgba(255, 255, 255, 0.16) 75%,transparent 75%,transparent);
}.game {display: flex;flex-direction: column;align-items: center;justify-content: center;width: 100%;height: 100%;background-color: #f0f0f0;.score {font-size: 30px;margin-bottom: 20px;}.speed {font-size: 30px;margin-bottom: 20px;}
}
.pause-bg {background-color: rgba(0, 0, 0, 0.5);z-index: 2;
}
</style>

这篇关于vue3,贪吃蛇的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1136581

相关文章

部署Vue项目到服务器后404错误的原因及解决方案

《部署Vue项目到服务器后404错误的原因及解决方案》文章介绍了Vue项目部署步骤以及404错误的解决方案,部署步骤包括构建项目、上传文件、配置Web服务器、重启Nginx和访问域名,404错误通常是... 目录一、vue项目部署步骤二、404错误原因及解决方案错误场景原因分析解决方案一、Vue项目部署步骤

前端原生js实现拖拽排课效果实例

《前端原生js实现拖拽排课效果实例》:本文主要介绍如何实现一个简单的课程表拖拽功能,通过HTML、CSS和JavaScript的配合,我们实现了课程项的拖拽、放置和显示功能,文中通过实例代码介绍的... 目录1. 效果展示2. 效果分析2.1 关键点2.2 实现方法3. 代码实现3.1 html部分3.2

CSS弹性布局常用设置方式

《CSS弹性布局常用设置方式》文章总结了CSS布局与样式的常用属性和技巧,包括视口单位、弹性盒子布局、浮动元素、背景和边框样式、文本和阴影效果、溢出隐藏、定位以及背景渐变等,通过这些技巧,可以实现复杂... 一、单位元素vm 1vm 为视口的1%vh 视口高的1%vmin 参照长边vmax 参照长边re

CSS3中使用flex和grid实现等高元素布局的示例代码

《CSS3中使用flex和grid实现等高元素布局的示例代码》:本文主要介绍了使用CSS3中的Flexbox和Grid布局实现等高元素布局的方法,通过简单的两列实现、每行放置3列以及全部代码的展示,展示了这两种布局方式的实现细节和效果,详细内容请阅读本文,希望能对你有所帮助... 过往的实现方法是使用浮动加

css渐变色背景|<gradient示例详解

《css渐变色背景|<gradient示例详解》CSS渐变是一种从一种颜色平滑过渡到另一种颜色的效果,可以作为元素的背景,它包括线性渐变、径向渐变和锥形渐变,本文介绍css渐变色背景|<gradien... 使用渐变色作为背景可以直接将渐China编程变色用作元素的背景,可以看做是一种特殊的背景图片。(是作为背

CSS自定义浏览器滚动条样式完整代码

《CSS自定义浏览器滚动条样式完整代码》:本文主要介绍了如何使用CSS自定义浏览器滚动条的样式,包括隐藏滚动条的角落、设置滚动条的基本样式、轨道样式和滑块样式,并提供了完整的CSS代码示例,通过这些技巧,你可以为你的网站添加个性化的滚动条样式,从而提升用户体验,详细内容请阅读本文,希望能对你有所帮助...

css实现图片旋转功能

《css实现图片旋转功能》:本文主要介绍了四种CSS变换效果:图片旋转90度、水平翻转、垂直翻转,并附带了相应的代码示例,详细内容请阅读本文,希望能对你有所帮助... 一 css实现图片旋转90度.icon{ -moz-transform:rotate(-90deg); -webkit-transfo

vue基于ElementUI动态设置表格高度的3种方法

《vue基于ElementUI动态设置表格高度的3种方法》ElementUI+vue动态设置表格高度的几种方法,抛砖引玉,还有其它方法动态设置表格高度,大家可以开动脑筋... 方法一、css + js的形式这个方法需要在表格外层设置一个div,原理是将表格的高度设置成外层div的高度,所以外层的div需要

Vue项目中Element UI组件未注册的问题原因及解决方法

《Vue项目中ElementUI组件未注册的问题原因及解决方法》在Vue项目中使用ElementUI组件库时,开发者可能会遇到一些常见问题,例如组件未正确注册导致的警告或错误,本文将详细探讨这些问题... 目录引言一、问题背景1.1 错误信息分析1.2 问题原因二、解决方法2.1 全局引入 Element

详解如何在React中执行条件渲染

《详解如何在React中执行条件渲染》在现代Web开发中,React作为一种流行的JavaScript库,为开发者提供了一种高效构建用户界面的方式,条件渲染是React中的一个关键概念,本文将深入探讨... 目录引言什么是条件渲染?基础示例使用逻辑与运算符(&&)使用条件语句列表中的条件渲染总结引言在现代