游戏设计:
- 规划游戏的核心功能,如场景、随机出现的地鼠、计分系统、游戏时间限制等。
- 简单设计游戏流程,包括开始界面、游戏进行中、关卡设置(如不同关卡地鼠出现数量、游戏时间等)、关卡闯关成功|失败、游戏结束闯关成功|失败等状态。
- 确定游戏的交互方式,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>
图片资源: