vue脚手架开发打地鼠游戏
- 其他
- 2025-09-02 05:21:01

游戏设计: 规划游戏的核心功能,如场景、随机出现的地鼠、计分系统、游戏时间限制等。简单设计游戏流程,包括开始界面、游戏进行中、关卡设置(如不同关卡地鼠出现数量、游戏时间等)、关卡闯关成功|失败、游戏结束闯关成功|失败等状态。确定游戏的交互方式,PC端测试鼠标左键点击击打地鼠,移动端手指点击击打地鼠。
以下为游戏开发前制作的游戏界面展示效果,如图: 其中有加分、减分对应分值的地鼠图片元素,洞口图片元素,以及使用CSS遮罩效果实现的地鼠出洞时像钻出来的效果图片元素(红色的),这里遮罩图片顶部切片向上延伸39px
问题解决:与游戏引擎开发不同,需要解决的问题如下:
1、html+css开发中,元素层级问题,很难直接实现地鼠从洞中钻出的效果这里使用CSS遮罩效果实现的地鼠出洞时像钻出来的效果,遮罩图片元素可见区域则是地鼠运动过程中可见区域,在此之外则不可见 以下为:相关CSS设置代码截图,需要注意的是:遮罩图片不可跨域使用,这里将图片文件转成Base64格式图片了,如图:
2、音频播放会有兼容性问题比如打到地鼠音效+加分或减分时,部分设备可能只听到一个音频;另外设置多次播放同一个音频时,会等一个播放结束后停顿后再重新播放。因此本游戏音效播放使用了Howler.js HTML5声音引擎,同一音频就可以重叠播放了。 Howler.js HTML5声音引擎 代码如下:
var rightMusic = new Howl({ src: ['static/right.mp3'], }); var wrongMusic = new Howl({ src: ['static/wrong.mp3'], }); var scoreAddMusic = new Howl({ src: ['static/scoreAdd.mp3'], }); var scoreReduceMusic = new Howl({ src: ['static/scoreReduce.mp3'], }); 实现游戏功能及游戏逻辑解读: 游戏逻辑代码:开始页(父组件): 包括开始页、游戏结束成功通关页、游戏结束失败未通关页 功能:触发开始闯关、闯关游戏结束获取所有关卡游戏得分数据渲染展示 游戏组件(子组件): 包括游戏页、关卡结束闯关成功页、关卡结束闯关失败页 功能:游戏交互逻辑代码
游戏资源准备:如音效(是否打到地鼠、加分、减分)、游戏场景图片、洞口图片、不同分值的地鼠、锤子图片等,如图:
速度控制-地鼠出洞/进洞:地鼠出洞/进洞动画时长可配置化处理,如图:
地鼠的随机出现和位置变化逻辑如图:
计分系统:计算、更新分数如图:
关卡难度变化及游戏时间的控制如图:
开始界面和结束界面的显示逻辑如图:
地鼠被打效果根据以上逻辑渲染游戏画面,锤子敲打地鼠,地鼠出洞/进洞,地鼠被打,如图:
效果展示:以下为:游戏主页面 | 游戏3关对应的游戏展示界面及加分、减分、闯关成功 | 闯关失败 | 通关失败 | 通关成功 截图
打地鼠通关录屏
打地鼠通关录屏
打地鼠未通关录屏
打地鼠未通关录屏
代码: 父组件代码: <template> <div> <!-- 首页 --> <div class="page index"> <button @click="startGame" class="index_btn">开始游戏</button> </div> <!-- 游戏页 --> <game ref="gameTemp" @gameMounted="gameLoaded" @gameOver="gameOverEnd"></game> <!-- 游戏结束 --> <div v-if="popIndex == 1" class="page pop"> <div class="pop_body"> <div class="end_body" :class="{ 'success':levelScoreData[levelScoreData.length - 1].currScore >= levelScoreData[levelScoreData.length - 1].targetScore, 'fail':levelScoreData[levelScoreData.length - 1].currScore < levelScoreData[levelScoreData.length - 1].targetScore, }"> <!-- 成功 --> <div v-if=" levelScoreData[levelScoreData.length - 1].currScore >= levelScoreData[levelScoreData.length - 1].targetScore " class="end_tips"> <p>恭喜您游戏通关啦</p> </div> <!-- 失败 --> <div v-else class="end_tips"> <p>游戏未通关哦~</p> </div> <!-- 所有关卡游戏得分数据 --> <div class="end_score_body"> <div v-for="(item,index) in levelScoreData" :key="'levelScoreData' + index" class="end_score_list"> <div>第{{index+1}}关</div> <div>关卡得分:{{item.currScore}}</div> <div>目标得分:{{item.targetScore}}</div> </div> </div> <!-- 继续游戏 --> <button class="end_btn end_btn1" @click="againGame">继续游戏</button> <!-- 关闭 --> <button class="end_btn end_btn2" @click="hideGameOver">关闭</button> </div> </div> </div> </div> </template> <script> export default { name: 'index', components:{ game:()=>import("@/views/game") }, data() { return { popIndex:0, // 1:游戏结束 levelScoreData:[], // 所有关卡游戏得分数据 } }, created(){ }, mounted(){ }, watch: { }, methods:{ // 游戏组件加载完毕 gameLoaded(){ // 开始闯关 // this.$refs.gameTemp.gameRun(); }, // 当前关卡闯关游戏结束 gameOverEnd(levelScoreData){ this.levelScoreData = levelScoreData; this.popIndex = 1; }, // popClose(){ // btnClickDo(()=>{ // if(this.popIndex == 1){ // this.levelScoreData = []; // this.$refs.gameTemp.showGame = false; // } // this.$nextTick(()=>{ // this.popIndex = 0; // }) // }) // }, // 首页-开始游戏 startGame(){ btnClickDo('.index_btn',()=>{ this.$refs.gameTemp.gameRun(); this.popIndex = 0; this.levelScoreData = []; }) }, // 游戏结束-继续游戏 againGame(){ btnClickDo('.end_btn1',()=>{ this.startGame(); }) }, // 游戏结束-关闭 hideGameOver(){ btnClickDo('.end_btn2',()=>{ this.popIndex = 0; this.levelScoreData = []; }) }, } } </script> <style scoped> .page{ width:100vw; height:100vh; position:fixed; left:0; top:0; overflow: hidden;} /* 首页 */ .page.index{ background-color: #fff; display: flex; justify-content: center; align-items: center; } .index_btn{ width: 300px; height: 80px; font-size: 30px; border-radius: 40px; border: none;} /* 弹层 */ .page.pop{ background-color: rgba(0,0,0,.5); padding-bottom: 100px; display: flex; justify-content: center; align-items: center; } .pop_body{ position: relative;} /* 游戏结束 */ .end_body{ width: 600px; padding: 80px 20px; border-radius: 20px; background-color: #fff;} .end_body.success{} .end_body.fail{} .end_tips{ padding-bottom: 40px; text-align: center;} .end_tips p{ line-height: 76px; font-size: 36px; font-weight: bold;} .end_score_body{ border: #999 solid 1px;} .end_score_list{ line-height: 60px; border-top: #999 solid 1px; display: flex; justify-content: space-between; align-items: center; } .end_score_list:first-child{ border-top: transparent;} .end_score_list div{ padding-left: 10px;} .end_score_list div:nth-child(1){ width: 20%;} .end_score_list div:nth-child(2){ width: 40%; border-left: #999 solid 1px; border-right: #999 solid 1px;} .end_score_list div:nth-child(3){ width: 40%;} .end_btn{ display: block; width: 370px; height: 80px; margin: 35px auto 0 auto; color: #fff; font-size: 30px; border-radius: 40px; border: none;} .end_body.success .end_btn{ background-color: green;} .end_body.fail .end_btn{ background-color: red;} </style> 子组件代码:game.vue <template> <div> <!-- 游戏页 --> <div v-show="showGame" class="page game"> <div class="game_body"> <!-- 游戏展示区 --> <div class="show_list_body"> <!-- 所有洞口 --> <div v-for="(item,index) in gameLevel[gameLevelIndex].num" :key="'all' + index" class="show_list"> <!-- CSS遮罩处理地鼠出洞效果 --> <div @click="wrongMusicPlay" class="show_list_mole"> <!-- 出洞地鼠 --> <img @click.stop="addScore(iidex,index)" v-for="(iitem,iidex) in gameImgList" :key="'imgBefore' + iidex" v-if="iitem.index == index && addScoreIndex !== index" :src="iitem.img" :style="'animation: fadeToTopTan ' + moleAnimationTime.outExecutionTime + 's ease both , fadeToDownHide ' + moleAnimationTime.enterExecutionTime + 's ' + moleAnimationTime.outExecutionTime + 's ease forwards;'" /> <!-- 被打地鼠 --> <img v-for="(iitem,iidex) in gameImgList" :key="'imgAfter' + iidex" v-if="iitem.index == index && addScoreIndex === index" :src="iitem.img" :style="'animation: beingBeaten .3s ease both , fadeToDownHide .2s .3s ease forwards;'" /> </div> <!-- 锤子-敲打 --> <img v-show="addScoreIndex === index" class="show_list_hammer" src="@/assets/img/game/hammer.png" /> </div> </div> <!-- <div v-if="!inGame" @click="gameStart" class="game_start_btn">开始游戏</div> --> <div class="show_time"> <div class="show_time_li"> <div>得分:<span><i>{{currScore}}</i></span></div> <div>目标:<span><i>{{gameLevel[gameLevelIndex].targetScore}}</i></span></div> </div> <div class="show_time_li"> <div>时间:<span><i>{{countdownTime}}</i>s</span></div> <div>关卡:<span><i>{{gameLevelIndex+1}}</i></span></div> </div> </div> <!-- 当前关卡得分分值集合 --> <div class="show_score"> <div v-for="(item,index) in currScoreData" :key="'score' + index" class="show_score_num">{{item > 0 ? '+' : ''}}{{item}}</div> </div> </div> </div> <!-- 关卡结束 --> <div v-show="showLevelEnd" class="page level_end"> <div class="level_end_body"> <!-- 当前关卡闯关成功 --> <div v-if="currScore >= gameLevel[gameLevelIndex].targetScore" class="level_end_success"> <!-- 非最后一关闯关成功 --> <div v-if="gameLevelIndex < gameLevel.length - 1"> <div class="level_end_title">恭喜您,本关卡闯关成功</div> <div @click="nextLevel" class="level_end_btn">下一关</div> </div> <!-- 最后一关闯关成功 --> <div v-else> <div class="level_end_title">恭喜您,本关卡闯关成功,已通过全部关卡</div> <div @click="nextLevel" class="level_end_btn">结束游戏</div> </div> </div> <!-- 当前关卡闯关失败 --> <div v-else class="level_end_fail"> <div class="level_end_title">很遗憾,本关卡未闯关成功</div> <div @click="again" class="level_end_btn">再试试</div> <div @click="over" class="level_end_btn_over">结束游戏</div> </div> </div> </div> </div> </template> <script> export default { components:{ }, data(){ return{ showGame:false, // 显示游戏页 inGame:false, // 是否游戏进行中 currScore:0, // 当前分值 currScoreData:[], // 当前关卡分值集合 addScoreIndex:'', // 哪个洞口地鼠被打到了 addScoreIndexArr:[], // 数组数据存储哪些洞口地鼠被打到了,主要用于处理每次出洞地鼠大于1个时,被打过的地鼠再次被打时导致的加分减分问题 countdownTiming:0, // countdownTimeDefault:30, // 初始化倒计时时间(秒) countdownTime:0, gameImgList:[], // 出洞地鼠-列表数据 // 地鼠图片-配置数据(图片及对应分值),游戏时从中随机取数据追加至gameImgList中 gameImgData:[ { img:require('@/assets/img/game/1.png'), score:1, }, { img:require('@/assets/img/game/2.png'), score:2, }, { img:require('@/assets/img/game/3.png'), score:3, }, { img:require('@/assets/img/game/4.png'), score:-1, }, // 炸弹-负数分值,如不需要去掉即可 ], // 地鼠出洞/进洞动画时长配置(控制 地鼠出洞/进洞 速度) moleAnimationTime:{ // outExecutionTime:.5, // 出洞动画执行时长 // enterExecutionTime:.3, // 进洞动画执行时长 outExecutionTime:.6, // 出洞动画执行时长 enterExecutionTime:.6, // 进洞动画执行时长 }, // 游戏所有关卡数据配置,如下3关:当前关卡的洞口数量、每次几个地鼠出洞、目标分值 gameLevel:[ { num:9, // 洞口数量 moleNum:1, // 每次几个地鼠出洞 targetScore:15, // 目标分值 countdownTimeDefault:20, // 倒计时时间(秒) }, { num:12, // 洞口数量 moleNum:2, // 每次几个地鼠出洞 targetScore:30, // 目标分值 countdownTimeDefault:40, // 倒计时时间(秒) }, { num:15, // 洞口数量 moleNum:3, // 每次几个地鼠出洞 targetScore:45, // 目标分值 countdownTimeDefault:60, // 倒计时时间(秒) }, ], gameLevelIndex:0, // 当前关卡(从0开始) // 关卡结束 showLevelEnd:false, } }, created() { }, mounted() { // this.gameRun(); this.$emit('gameMounted'); }, watch:{ }, methods:{ // 开始游戏,计时等设置 startGame(){ this.inGame = true; this.currScore = 0; this.currScoreData = []; this.setGameInit(); this.countdownTiming = 0; // this.countdownTime = this.countdownTimeDefault; this.countdownTime = this.gameLevel[this.gameLevelIndex].countdownTimeDefault; this.changeTime(); }, // 计时 // timing , rafId; changeTime(k){ // console.log(k); if(!this.timing && k){ this.timing = k } // 1秒执行60次 this.rafId = requestAnimationFrame(this.changeTime); // 倒计时计算 this.countdownTiming++; // 1秒(1000ms)执行一次 if(this.countdownTiming % 60 == 0){ this.countdownTime-= 1; } if(this.countdownTime <= 0){ // 关卡结束 this.showLevelEnd = true; cancelAnimationFrame(this.rafId); clearTimeout(this.timer); } }, // 动态设置 出洞地鼠-列表数据(设置随机洞口出现) setGameInit(){ this.addScoreIndexArr = []; this.addScoreIndex = ''; let currLevelNum = this.gameLevel[this.gameLevelIndex].num; // 页面中呈现的所有洞口KEY集合 let randomLevelKey = []; for(var i=0; i < currLevelNum; i++){ randomLevelKey.push(i); } // 页面中呈现的所有洞口KEY集合,打乱顺序 randomLevelKey = randomLevelKey.sort(function(a, b){return 0.5 - Math.random();}); // console.log(randomLevelKey); let moleNum = this.gameLevel[this.gameLevelIndex].moleNum; this.gameImgList = []; // 解决与上次同一洞口导致不出现问题 this.$nextTick(()=>{ for(var i=0; i < moleNum; i++){ // index 出洞地鼠展示在对应KEY的洞口(这样设置保证KEY不会重复) this.gameImgList.push({index:randomLevelKey[i],...this.gameImgData[Math.floor(Math.random() * this.gameImgData.length)]}); } // 不断展示随机出现的地鼠定时器 this.timer = setTimeout(()=>{ this.setGameInit(); // },700) // 根据 地鼠出洞/进洞动画时长配置 计算设置 },(this.moleAnimationTime.enterExecutionTime + this.moleAnimationTime.outExecutionTime) * 1000 - 100) }) }, /// gameRun(){ this.gameLevelIndex = 0; if(this.showGame){ this.startGame(); }else{ this.showGame = true; this.$nextTick(()=>{ this.$nextTick(()=>{ this.startGame(); }) }) } }, // 开始游戏 // gameStart(){ // this.gameLevelIndex = 0; // this.startGame(); // }, // 计时结束-游戏结束 gameEnd(){ // console.log(this.currScore); // console.log(this.currScoreData); this.inGame = false; this.$emit('gameOver',this.levelScoreData); }, /// // 加分减分统计 addScore(index,addScoreIndex){ // 解决快速点击同一个地鼠不停计算分值问题(且处理每次出洞地鼠大于1个时,被打过的地鼠再次被打时导致的加分减分问题) if(this.addScoreIndexArr.indexOf(addScoreIndex) != -1){ return; } this.addScoreIndexArr.push(addScoreIndex); this.addScoreIndex = addScoreIndex; // setTimeout(()=>{ // this.addScoreIndexArr = []; // this.addScoreIndex = ''; // },500) // 打到地鼠音效 rightMusic.play(); let score = this.gameImgList[index].score; if(score > 0){ // 加分对应音效 scoreAddMusic.play(); }else{ // 减分对应音效 scoreReduceMusic.play(); } this.currScore += score; this.currScoreData.push(score); // console.log(this.currScore); // console.log(this.currScoreData); }, wrongMusicPlay(){ // 未打到地鼠音效 wrongMusic.play(); }, // 当前关卡闯关成功-下一关 nextLevel(){ btnClickDo('.level_end_btn',()=>{ this.setLevelScore(); this.showLevelEnd = false; if(this.gameLevelIndex >= (this.gameLevel.length - 1)){ // 所有关卡结束 this.gameEnd(); this.showGame = false; }else{ // 下一关 this.gameLevelIndex++; this.startGame(); } }) }, // 当前关卡闯关失败-再试试 again(){ btnClickDo('.level_end_btn',()=>{ this.showLevelEnd = false; this.startGame(); }) }, // 当前关卡闯关失败-结束游戏 over(){ btnClickDo('.level_end_btn_over',()=>{ this.setLevelScore(); this.showLevelEnd = false; // 游戏结束 - 闯关失败-结束游戏 this.gameEnd(); this.showGame = false; }) }, // 每一关结束存储当前关卡游戏得分数据 setLevelScore(){ if(this.gameLevelIndex == 0){ this.levelScoreData = []; this.levelScoreData.push({ targetScore:this.gameLevel[this.gameLevelIndex].targetScore, currScore:this.currScore, }); }else{ this.levelScoreData.push({ targetScore:this.gameLevel[this.gameLevelIndex].targetScore, currScore:this.currScore, }); } }, } } </script> <style> /* 地鼠出洞 */ @keyframes fadeToTopTan{ 0%{ transform:translate(0,100%) scale(1,1) rotateY(0); opacity:0;} 70%{ transform:translate(0,0) scale(1,1.1) rotateY(0); opacity:1;} 100%{ transform:translate(0,0) scale(1,1) rotateY(0); opacity:1;} } /* 地鼠进洞 */ @keyframes fadeToDownHide{ 0%{ transform:translate(0,0) scale(1,1) rotateY(0); opacity:1;} 100%{ transform:translate(0,100%) scale(1,1) rotateY(0); opacity:0;} } /* 地鼠被打 */ @keyframes beingBeaten{ 0% , 10% ,30% , 50% , 100%{ transform:translate(0,0) scale(1,1) rotateY(0); opacity:1;} 20% , 40% , 60%{ transform:translate(8px,0) scale(1,1) rotateY(20deg); opacity:1;} } </style> <style scoped> .page{ width:100%; height:100%; position:absolute; left:0; top:0; overflow: hidden;} /* 游戏页 */ .page.game{ overflow-y: auto; -webkit-overflow-scrolling: touch;} .game_body{ min-height: 100vh; padding-top: calc(10px * 2 + 130px); background: url(../assets/img/game/bg.png) no-repeat center top; background-size: 100%; overflow: hidden;} .show_time{ width: calc(100vw - 20px); height: 130px; padding: 20px; background-color: #fff; position: absolute; left: 10px; top: 10px; display: flex; justify-content: space-between; align-items: center; } .show_time_li{} .show_time_li div{ display: flex; justify-content: flex-start; align-items: center; } .show_time_li div span{ width: 50px; white-space: nowrap;} /* 当前关卡分值集合 */ .show_score{ width: 100%; position: fixed; left: 0; top: 180px; pointer-events: none;} .show_score_num{ width: 100%; text-align: center; color: #fff; font-size: 80px; font-weight: bold; position: absolute; left: 0; top: 0; text-shadow: #fc6100 4px 4px, #fc6100 4px -4px, #fc6100 -4px 4px, #fc6100 -4px -4px; animation: scoreHide .5s .1s linear forwards; } @keyframes scoreHide{ 0%{ transform:translateY(0); opacity:1;} 100%{ transform:translateY(-100%); opacity:0;} } /* 游戏展示区 */ .show_list_body{ /* padding: 0 10px; */ min-height: calc(100vh - (10px * 2 + 130px)); max-height: calc(243px * 5 + 20px * 4); display: flex; flex-direction: row; flex-wrap: wrap; justify-content: space-around; align-items: center; align-content: space-around; } /* 所有洞口 */ .show_list{ width: 235px; height: 243px; background: url(../assets/img/game/list_bg.png); background-size: 100% 100%; position: relative;} /* .show_list:nth-child(3) ~ .show_list{ margin-top: 20px;} */ /* CSS遮罩处理地鼠出洞效果 */ .show_list_mole{ width: 235px; height: 282px; padding-top: 39px; position: absolute; left: 0; bottom: 0; -webkit-mask: url() repeat center top; -webkit-mask-size: 100% 100%; } /* 出洞地鼠 */ .show_list_mole img{ width: 100%; height: 100%; transform-origin: center 203px; } /* 锤子-敲打 */ .show_list_hammer{ width: 171px; position: absolute; left: 80px; top: -80px; animation: hammerStrike .3s ease both; } /* 锤子敲打 */ @keyframes hammerStrike{ 0%{ transform:translate(60px,-60px) rotate(15deg); opacity: 1;} 80%{ transform:translate(0,0) rotate(-15deg); opacity: 1;} 100%{ transform:translate(0,0) rotate(-15deg); opacity: 0;} } .game_start_btn{ padding: 0 15px; height: 60px; line-height: 60px; border-radius: 30px 0 0 30px; background-color: pink; position: absolute; right: 0; top: 60vh;} /* 关卡结束 */ .page.level_end{ background-color: rgba(0,0,0,.5); display: flex; justify-content: center; align-items: center; } .level_end_body{ width: 600px; padding: 80px 20px; border-radius: 20px; background-color: #fff;} /* 当前关卡闯关成功 */ .level_end_success{} /* 当前关卡闯关失败 */ .level_end_fail{} .level_end_title{ height: 60px; line-height: 60px; margin-bottom: 60px; text-align: center;} .level_end_btn, .level_end_btn_over{ width: 300px; height: 80px; line-height: 80px; margin: 0 auto; text-align: center; color: #fff; border-radius: 40px;} .level_end_success .level_end_btn{ background-color: green;} .level_end_fail .level_end_btn, .level_end_fail .level_end_btn_over{ background-color: red;} .level_end_btn_over{ margin-top: 35px;} </style> 图片资源:
vue脚手架开发打地鼠游戏由讯客互联其他栏目发布,感谢您对讯客互联的认可,以及对我们原创作品以及文章的青睐,非常欢迎各位朋友分享到个人网站或者朋友圈,但转载请说明文章出处“vue脚手架开发打地鼠游戏”
上一篇
数据库提权总结
下一篇
基于Cookie追踪用户行为