【整理】HTML5游戏开发学习笔记(4)- 记忆力游戏
1.预备知识
(1)Canvas绘制多边形
(2)Canvas绘制文字
2.实现思路
涉及的对象
(1)场景Scene
场景代表了画布上的一块区域,场景里的每个物体都是场景里的一个元素,其绘制统一由场景来调用绘制
(2) 扑克牌Card
包括翻开,关闭,移除等操作
(3)一副扑克牌Deck
包括洗牌
(4)游戏玩法PlayingRule
每次选择2张扑克进行比较,相等则消除(移除),不相等,则进行下一次的2张牌的选择 ,在进行比较
3.主要代码
/*场景*/
function Scene(canvasId){
var canvas = document.getElementById(canvasId)
var ctx = canvas.getContext('2d')
var width = canvas.width
var height = canvas.height //场景里所有的元素
var elements = [] function initEvents(){
canvas.addEventListener('click',function(e){
for(var i=0;i<elements.length;i++){
if( typeof elements[i].onEvent == 'function' ){
elements[i].onEvent(e)
}
}
},false)
} return { addElement : function(element){
elements.push(element)
}, init : function(){ for(var i=0;i<elements.length;i++){
if( typeof elements[i].init == 'function' ){
elements[i].init(ctx)
}
} initEvents()
} }
} /*
扑克牌
德国扑克牌(一付55/54/32张)规格:9x5.7cm 8.7x5.7cm
*/
function Card(x,y,value,index){
this.x = x
this.y = y
this.value = value
this.index = index
this.status = this.StatusType.Closed this.ctx = null
} Card.prototype = { constructor : Card,
width : 60,
height : 90,
ratio : 1,
StatusType : {Closed:0,Opened:1,Removed:2},
timeout : 300, init : function(ctx){
this.ctx = ctx
this.draw()
}, draw : function(){
var ctx = this.ctx
ctx.clearRect(this.x,this.y,this.width,this.height) switch(this.status){
case 2 :
ctx.fillStyle = '#ccc'
ctx.fillRect(this.x,this.y,this.width*this.ratio,this.height*this.ratio)
break
break
case 1 :
ctx.fillStyle = '#36f'
ctx.fillRect(this.x,this.y,this.width*this.ratio,this.height*this.ratio)
//添加文字
ctx.fillStyle = '#fff'
ctx.font = 'bold 50px Verdana'
ctx.fillText(this.value,this.x+this.width*.5-17,this.y+this.height*.5+17)
break
case 0 :
default :
ctx.fillStyle = '#f60'
ctx.fillRect(this.x,this.y,this.width*this.ratio,this.height*this.ratio)
break
}
}, onEvent : function(e,cardPicked){
var x = e.offsetX,
y = e.offsetY //牌被选中,显示该牌的value值
if(this.x<=x&&x<=this.x+this.width&&this.y<=y&&y<=this.y+this.height){
this.open()//翻开扑克 var self = this
cardPicked(self) setTimeout(function(){
self.close()//关闭扑克牌
},self.timeout) } cardPicked(null)
}, open : function(){
if(this.status == this.StatusType.Closed){
this.status = this.StatusType.Opened
this.draw()
}
}, close : function(){
if(this.status == this.StatusType.Opened){
this.status = this.StatusType.Closed
this.draw()
}
}, remove : function(){
this.status = this.StatusType.Removed
this.draw()
} } /*整付扑克牌 52张*/
function Deck(){
this.cards = []
this.rule = new PlayingRule()
} Deck.prototype = {
constructor : Deck,
opts : {},
//cards : 52,
margin : 10,//牌之间的间距 init : function(ctx){
this.opts.ctx = ctx this.draw()
this.shuffle()
}, draw : function(){
var ctx = this.opts.ctx
var x = this.margin,
y = this.margin,
i = 3,
index = -1 for(i=3;i<9;i++){
var card1 = new Card(x,y,i,++index)
var card2 = new Card(x,y+card1.height+this.margin,i,++index) card1.init(ctx)
card2.init(ctx) //console.log(card1.x+','+card1.y)
//console.log(card2.x+','+card2.y) this.cards.push(card1)
this.cards.push(card2) x += card1.width+this.margin
y = y
}
}, //洗牌
shuffle : function(){
var length = this.cards.length
var r1
var r2
var i
var value for(i=0;i<length;i++){
r1 = Math.floor(Math.random()*length)
r2 = Math.floor(Math.random()*length) value = this.cards[r1].value
this.cards[r1].value = this.cards[r2].value
this.cards[r2].value = value
} /*debug*/
for(i=0;i<length;i++){
console.log(this.cards[i].index+' : '+this.cards[i].value)
}
}, onEvent : function(e){
var cards = this.cards
var length = cards.length
var i
var card
var self = this for(i=0;i<length;i++){
card = cards[i] if(typeof card.onEvent == 'function'){
card.onEvent(e,function(card){
if(card==null||card.status==card.StatusType.Removed){
return
} //有扑克牌被选中
var isMatch = self.rule.pushCard(card).isMatch()
var cards = self.rule.getCards()
var length = cards.length if(isMatch){
for(var i=0;i<length;i++){
var card = cards[i]
alert(card.index)
card.remove()
}
}
else{
//如果不匹配,当前选中的扑克牌会自动定时关闭
//DONOTHING
} //如果有2张扑克,则清除玩法中的选中的2张扑克牌
self.rule.refresh()
})
}
}
} } /*玩法规则*/
function PlayingRule(){
var cards = [] return { pushCard : function(card){
if(cards.length<2){
cards.push(card) console.log('push:'+card.value)
} return this
}, isMatch : function(){
if(cards.length==2){
return cards[0].value === cards[1].value
} return false
}, refresh : function(){
if(cards.length==2){
cards.length = 0
}
}, getCards : function(){
return cards;
} }
} //app
window.onload = function(){ var scene = new Scene('canvas1')
var deck = new Deck() scene.addElement(deck)
scene.init() }
4.优化和完善
(1)有bug,可以作弊,选择的拥有比较的2张牌可以为同一张,需添加判断
(2)简化了书中的例子,图形图片绘制改为了数字文字的现实
(3)可以进行图形绘制,使扑克牌逼真些
(4)考虑自适应大小的设置,如扑克牌的比例随着手机屏幕大小自适应
【整理】HTML5游戏开发学习笔记(4)- 记忆力游戏的更多相关文章
- 【整理】HTML5游戏开发学习笔记(5)- 猜谜游戏
距上次学习笔记已有一个多月过去了,期间由于新项目赶进度,以致该学习计划给打断,十分惭愧.书本中的第六章的例子相对比较简单.所以很快就完成. 1.预备知识html5中video标签的熟悉 2.实现思路对 ...
- 【整理】HTML5游戏开发学习笔记(1)- 骰子游戏
<HTML5游戏开发>,该书出版于2011年,似乎有些老,可对于我这样没有开发过游戏的人来说,却比较有吸引力,选择自己感兴趣的方向来学习html5,css3,相信会事半功倍.不过值得注意的 ...
- HTML5移动开发学习笔记之Canvas基础
1.第一个Canvas程序 看的是HTML5移动开发即学即用这本书,首先学习Canvas基础,废话不多说,直接看第一个例子. 效果图为: 代码如下: <!DOCTYPE html> < ...
- [游戏开发-学习笔记]菜鸟慢慢飞(四)-Camera
游戏开发中,主相机应该是最重要的GameObject之一,毕竟游戏呈现给玩家,就是通过它. 相机的使用,在不同的游戏中,有很大的不同.这里总结一下自己学到的一些相关知识. 固定位置-游戏过程中相机的T ...
- cocos2d-x 3.x游戏开发学习笔记(1)--mac下配置cocos2d-x 3.x开发环境
打开用户文件夹下.bash_profile文件,配置环境 vim ~/.bash_profile //按键i,进行插入编辑(假设输错d进行删除一行) 环境配置过程例如以下: 1.首先配置下androi ...
- 【Visual C++】游戏编程学习笔记之三:游戏循环的使用
本系列文章由@二货梦想家张程 所写,转载请注明出处. 本文章链接:http://blog.csdn.net/terence1212/article/details/44208419 作者:Zee ...
- 【整理】HTML5游戏开发学习笔记(2)- 弹跳球
1.预备知识(1)在画布上绘制外部图片资源(2)梯度(gradient):HTML5中的对象类型,包括线性梯度和径向梯度.如createLinearGradient,绘制梯度需要颜色组http://w ...
- 【整理】HTML5游戏开发学习笔记(3)- 抛物线运动
1.预备知识(1)Canvas旋转的实现过程 window.onload = function(){ var ctx = document.getElementById('canvas1').getC ...
- [Android游戏开发学习笔记]View和SurfaceView
本文为阅读http://blog.csdn.net/xiaominghimi/article/details/6089594的笔记. 在Android游戏中充当主要角色的,除了控制类就是显示类.而在A ...
随机推荐
- 使用tomcat,不能连接localhost/8080的解决办法
首先,java的一些环境变量要解决. 其次,tomcat也应该各种环境变量设置好. 最后,把下图的那个地址重新选择一遍. 记住以上每一步弄好了之后都重启一下机器. 我也不知道为什么,但是有些就是从起之 ...
- PAT甲题题解-1008. Elevator (20)-大么个大水题,这也太小瞧我们做题者的智商了
如题... #include <iostream> #include <cstdio> #include <algorithm> #include <cstr ...
- PAT甲题题解-1106. Lowest Price in Supply Chain (25)-(dfs计算树的最小层数)
统计树的最小层数以及位于该层数上的叶子节点个数即可. 代码里建树我用了邻接链表的存储方式——链式前向星,不了解的可以参考,非常好用: http://www.cnblogs.com/chenxiwenr ...
- Daily Scrum NO.1
工作概况 符美潇(PM): 今日工作 1.根据开发进程分配第一步开发工作,对相应的成员提出今日的开发要求:要求成员自己所负责的线程池,动态爬取,去重,文件分类等部分进行资料的相关了解. 2.Daily ...
- linux内核分析--操作系统是如何工作的?
一个简单的时间片轮转多道程序 操作系统的"两把剑":中断上下文(保存现场和恢复现场)和进程上下文的切换 源代码的分析 *使用的源代码为视频中所使用的精简内核的源代码 首先分析myp ...
- week9:个人博客作业
团队作业(5) 以下内容多数是网上的内容,只是做了整合的过程. 要求 在PM 带领下, 每个团队深入分析下面行业的App, 找到行业的Top 5 (从下面的三个备选中,任选一个行业即可) 英语学习/词 ...
- [转帖]PG里面的Citus简介----找时间学习一下.
1. Citus是什么 是PostgreSQL的扩展,可以同PG一同安装,之后通过SQL命令加入到数据库中. [相关操作] ? 1 2 #创建Citus扩展: CREATE EXTENSION cit ...
- 深入解读Quartz的原理
Quartz是一个大名鼎鼎的Java版开源定时调度器,功能强悍,使用方便. 一.核心概念 Quartz的原理不是很复杂,只要搞明白几个概念,然后知道如何去启动和关闭一个调度程序即可. 1 ...
- SP1043 GSS1
题目链接 简单说就是带修的查询区间最大子段和,用线段树维护即可 对于每个区间,我们肯定要记录它的最大子段和\(v\),但是怎么维护呢? 我们可以记录下从区间左端点开始的最大子段和\(v1\),从右端点 ...
- poj 2482 Stars in Your Window + 51Nod1208(扫描线+离散化+线段树)
Stars in Your Window Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 13196 Accepted: ...