Python Pygame实战之红心大战游戏的实现

 更新时间:2022年02月09日 14:56:52   作者:木木子学python  
说起Windows自带的游戏,相信许多80、90后的朋友都不陌生。本文就将利用Python中的Pygame模块实现一下windows经典游戏之一的红心大战,需要的可以参考一下

导语

还记得那些年,我们玩过的Windows小游戏吗?

说起Windows自带的游戏,相信许多80、90后的朋友都不陌生。

在很早的那个游戏贫瘠的年代,《纸牌》、《扫雷》等游戏帮助我们在微机课上带来了许多欢乐的时光。但在这些游戏中,有一款游戏似乎玩懂的人,甚至知道玩法的人并不多。这款游戏就是《红心大战》。

最近小编心血来潮仔细钻研了一下之后,发现这款游戏玩起来却比其他的几款游戏更带感一些。因为这款游戏的玩法简单概括就是两个字:“坑人”,而且是明着坑的那种。

那大家猜到了哈——今天跟着木木子一起开启《红心大战》纸牌小游戏吧!

一、 红心大战用户手册

1.打开游戏:双击hongxindazhan.py 主程序运行即可!

2.开始游戏:键入玩家姓名,创建新ID,点确认进入游戏。

3.游戏流程:

1)换牌:选择三张手牌,点击上方按钮完成交换;

2)出牌:轮到玩家出牌时,选择手牌单击便可打出。(具体出牌规则详见附录)注:当此时机不能打出此牌时,下方提示栏会显示无法打出原因。

4.统分阶段:一轮游戏过后,会弹出得分表,显示玩家和三名电脑每轮积分和总积分,并显示玩家当前名次。

5.菜单栏:

1)游戏栏:新游戏(F2):单击开启新游戏。得分(F4):单击显示得分表。背景音乐:单击切换音乐开关。选项(F5):单击修改电脑出牌速度。退出:单击退出游戏。

2)帮助栏:规则介绍(F1):单击显示简要规则和获胜条件。名言:单击弹出一则名言。

6.背景音乐:更换:可用自己的wav格式音乐重命名为‘m1.wav’替换程序目录下的同名文件。

二、红心大战游戏规则

1、在玩游戏前,需要决定庄家。在电脑中,庄家坐南。

2、拿到一手牌(共计13张)后,庄家首先须选出三张牌传给其他对手。第一局把牌传给左手边的玩家;第二局把牌传给右手边的玩家;第三局把牌传给坐在对面的玩家;第四局不传牌,依此类推。接到庄家的传牌后,也需要任意传回给庄家三张牌。在电脑中,如要选牌,单击相应牌张即可。如要取消选定的牌,请再次单击。

3、抓有梅花2 的玩家必须首先出梅花2,谓首攻。

4、然后按顺时针方向出牌。每位玩家依次必须跟同花色牌。如果已经没有与发牌花色相同的牌,则可以出任何一张牌。唯一例外是不能在第一圈牌中出红桃或黑桃Q(通常称为“猪”)。注意:出的同一花色牌中最大的牌会赢取这一圈,赢牌的玩家在下一圈中先出牌。只有前面出过红桃以后,才可以拿红桃领出(除非手中只剩下红桃一种花色的牌张)。

5、每一轮游戏结束时,每张红心计1 分,“黑桃皇后(猪)”计13 分。游戏将持续到有人得100 分或更多分或者庄家退出游戏时结束。如果在一轮牌中赢得了所有的红心和“黑桃皇后”(称之为“全收”),则“全收”者得零分,其余玩家每人得26 分。该游戏的得分越低越好。

三、准备中

1)环境安装

环境安装 本文用到的运行环境:Python3.7、Pycharm社区版2020、tkinter模块、部分自带模块直接导入不需要安装。

模块安装:

pip install -i pypi.douban.com/simple/ +模块名 

2)背景选择最原始的绿色背景。(仅展示背景,其余图片比较多不展示)

四、代码演示

​仅展示主程序源码。每行代码都有注释所以直接展示!

from Tkinter import *
from tkFont import *
import winsound
from tkMessageBox import *
from inner import *
from dialog import *
#游戏主界面创建
class GameFrame:
    def __init__(self,game,bgimg):#game,Game类对象;bgimg,背景图片
        self.master = game.root
        #创建画布,绘制背景
        self.c = Canvas(self.master,width = 1024,height = 640)
        self.c.create_image(514,322,image=bgimg)
        self.c.pack()
        #创建状态条
        self.status = Label(self.master,text=" 欢迎参加红心大战!",
                            bd=1,relief=SUNKEN,anchor=W)
        self.status.pack(fill = X)
        #打开开始对话框
        startdialog = StartDialog(self.master,"红心大战")
        if startdialog.isCancel:
            game.cancel()
        else:
            self.gamemodetext = ['向左传','向右传','交叉换牌']
            self.name = [startdialog.name,'西','北','东']
            self.handXY = [[346.5,490,1,0],[20,135,0,1],
                           [586.5,20,-1,0],[913,375,0,-1]]
            self.nameXY = [[-20,130,SE],[0,-20,SW],[91+20,0,NW],[91,130+20,NE]]
            self.middleXY = [[466.5,330],[411,255],[466.5,180],[522,255]]
            self.img = 53 * ['']
            for i in range(52):
                self.img[i] = PhotoImage(file = 'card\\%s.pgm' % (i))
            self.img[52] = PhotoImage(file = 'back.pgm')
            
            self.scorelist = []
            self.cards = []
            self.gamemode = 0
            self.speed = 100
            self.wait = self.speed
            self.isChanging = False
            self.iswait = False
            #绘制名字
            for i in range(4):
                self.c.create_text(self.handXY[i][0]+self.nameXY[i][0],
                                   self.handXY[i][1]+self.nameXY[i][1],
                                   fill = 'white',text = self.name[i],
                                   anchor = self.nameXY[i][2],
                                   font = Font(size=15,weight="bold"))
            #创建手牌
            self.l = 52 * ['']
            for i in range(52):
                self.l[i] = Label(self.master,image=self.img[52],bd = -1)
            for i in range(13):
                self.l[i]['text'] = str(i)
                self.l[i].bind("<Button-1>",self.cardEvent)
            #创建中央牌
            self.ml = 4 * ['']
            for i in range(4):
                self.ml[i] = Label(self.master,image=self.img[52],bd = -1)
            
            self.b = Button(self.master,width=15,command=self.buttonEvent)
            #一轮游戏开始
            self.oneGameStart()
    def oneGameStart(self):
        #创建Onegame对象,用来获取出牌信息
        self.onegame = OneGame()
        #获取玩家手牌,显示
        hand = self.onegame.getPlayerHand(0)
        for i in range(13):
            self.l[i]['image'] = self.img[hand[i].id]
        for i in range(4):
            for j in range(13):
                self.moveCard(i,j,0)
        #游戏模式为0,1,2时进入换牌阶段        
        if self.gamemode != 3:
            self.changeHands()
        else:
            self.onegame.changeCards([],3)
            self.isChanging = False
            self.leftCards = 13
            #进入出牌阶段,玩家前的电脑出牌
            self.playpreCards()
    def changeHands(self):
        #换牌阶段相应初始化
        self.select = []#所选的牌
        self.isChanging = True
        self.isOK = False
        s = [1,3,2]
        self.status['text'] = ' 请选三张牌传给'+self.name[s[self.gamemode]]+'。'
        #提示按钮显示
        self.b['text'] = self.gamemodetext[self.gamemode]
        self.b.place(x = 460,y = 400)
        self.b['state'] = DISABLED
    def cardEvent(self,event):
        #牌的事件
        #获取牌的位置
        i = int(event.widget['text'])
        if not self.isChanging:
            #出牌阶段的牌事件
            #等待中不出牌
            if self.iswait:
                return
            #出不了的牌不出
            if not self.onegame.available(i):
                self.status['text'] = self.onegame.errorString
                return
            self.onegame.playCard(i)
            #所出牌显示到中央
            event.widget.place_forget()
            self.ml[self.turn]['image'] = event.widget['image']
            self.ml[self.turn].place(x = self.middleXY[0][0],y = self.middleXY[0][1])
            self.turn += 1
            self.leftCards -= 1
            #玩家后的电脑出牌
            self.iswait = True
            self.playlaterCards()
            self.status['text'] = ' 正在等候……'
            self.wait += 500 + 5 * self.speed
            #等待一段时间后,中央牌清空,玩家前的电脑出牌
            self.master.after(self.wait,self.playpreCards)
            self.wait = self.speed
        else:
            #换牌阶段的牌事件,弹起的落下,落下的弹起
            if not self.isOK:
                if i in self.select:
                    self.select.remove(i)
                    self.moveCard(0,i,0)
                    self.b['state'] = DISABLED
                else:
                    if len(self.select) < 3:
                        self.select.append(i)
                        self.moveCard(0,i,1)
                        if len(self.select) == 3:
                            self.b['state'] = NORMAL
    def buttonEvent(self):
        #提示按钮事件
        if not self.isOK:
            #换牌前获取电脑换的牌进行交换
            self.select = self.onegame.changeCards(self.select,self.gamemode)
            hand = self.onegame.getPlayerHand(0)
            for i in range(13):
                self.l[i]['image'] = self.img[hand[i].id]
                self.moveCard(0,i,0)
            for i in self.select:
                self.moveCard(0,i,1)
            self.status['text'] = ' 请按"确定"接受传来的牌。'
            self.b['text'] = '确定'
            self.isOK = True
        else:
            #换牌后进行确认,进入出牌阶段,玩家前的电脑出牌
            for i in self.select:
                self.moveCard(0,i,0)
            self.b.place_forget()
            self.isChanging = False
            self.leftCards = 13
            self.playpreCards()
    def playpreCards(self):
        if self.leftCards == 0:
            #牌出完了进行统分,显示得分对话框
            for i in range(4):
                hand = self.onegame.p[i].scoreHand
                for j in range(len(hand)):
                    self.l[i*13+j]['image'] = self.img[hand[j].id]
                    self.moveCard(i,j,0)
            score = self.onegame.getScore()
            self.scorelist.append(score)
            if len(self.scorelist) != 1:
                for i in range(4):
                    self.scorelist[-1][i] += self.scorelist[-2][i]
            self.status['text'] = ' 得分'
            scoredialog = self.showScoreDialog()
            #得分确认后初始化,开始新一轮游戏
            if scoredialog.isover:
                self.scorelist = []
                self.gamemode = 0
            else:
                self.gamemode = (self.gamemode + 1) % 4
            for i in range(52):
                self.l[i].place_forget()
                self.l[i]['image'] = self.img[52]
            for i in range(4):
                self.ml[i].place_forget()
            self.oneGameStart()
        else:
            #获取玩家之前的电脑出牌,进行显示
            for i in range(4):
                self.ml[i].place_forget()
            self.turn = 0
            preCards = self.onegame.preCard
            if len(preCards) == 0:
                self.end()
            else:
                for i in range(len(preCards)):
                    p,j = preCards[i][0],preCards[i][1]
                    self.cards.append([self.turn,p,j])
                    self.master.after(self.wait,self.showMiddleCard)
                    if p == 3:
                        self.master.after(self.wait,self.end)
                        self.wait = self.speed
                    else:
                        self.wait += self.speed
                    self.turn += 1
    def playlaterCards(self):
        #获取玩家之后的电脑出牌,进行显示
        laterCards = self.onegame.laterCard
        for i in range(len(laterCards)):
            p,j = laterCards[i][0],laterCards[i][1]
            self.cards.append([self.turn,p,j])
            self.master.after(self.wait,self.showMiddleCard)
            self.wait += self.speed
            self.turn += 1
    def moveCard(self,i,j,state):#i,玩家;j,第几张牌;state,弹起还是放下
        #牌的移动
        self.l[i*13+j].place_forget()
        x0 = self.handXY[i][0] + self.handXY[i][2]*j*20
        y0 = self.handXY[i][1] + self.handXY[i][3]*j*20
        self.l[i*13+j].place(x = x0,y = y0 - state * 20)
    def newGame(self):
        #新游戏
        self.scorelist = []
        self.gamemode = 0
        for i in range(52):
                self.l[i].place_forget()
                self.l[i]['image'] = self.img[52]
        for i in range(4):
            self.ml[i].place_forget()
        self.oneGameStart()
    def showScoreDialog(self):
        #显示分数对话框
        scoredialog = ScoreDialog(self.master,self.scorelist,self.name)
        return scoredialog
    def showMiddleCard(self):
        #中央牌显示
        i = self.cards[0][0]
        p = self.cards[0][1]
        j = self.cards[0][2]
        del self.cards[0]
        self.l[p*13+j].place_forget()
        hand = self.onegame.getPlayerHand(p)
        self.ml[i]['image'] = self.img[hand[j].id]
        self.ml[i].place(x = self.middleXY[p][0],
                         y = self.middleXY[p][1])
    def end(self):
        #电脑出完牌至玩家出牌的切换
        self.iswait = False
        self.status['text'] = ' 请出一张牌。'

#游戏窗口创建,并建立游戏菜单项
class Game:
    def __init__(self):
        #建立根窗口,设置
        self.root = Tk()
        self.root.title("红心大战")
        self.root.geometry('+150+10')
        self.root.resizable(False, False)
        #播放背景音乐
        self.s = winsound.PlaySound('m1.wav',
                                    winsound.SND_ASYNC+winsound.SND_LOOP)
        #建立菜单
        m = Menu(self.root)
        self.root['menu'] = m
        gamemenu = Menu(m)
        helpmenu = Menu(m)
        m.add_cascade(label = '游戏',menu = gamemenu)
        m.add_cascade(label = '帮助',menu = helpmenu)
        gamemenu.add_command(label="新游戏     F2",command = self.gameEvent1)
        gamemenu.add_separator()
        gamemenu.add_command(label="得分...    F4",command = self.gameEvent2)
        gamemenu.add_command(label="选项...    F5",command = self.gameEvent4)
        self.v = IntVar()
        self.v.set(1)
        gamemenu.add_checkbutton(label="背景音乐",variable = self.v,
                                 command = self.gameEvent3)
        gamemenu.add_separator()
        gamemenu.add_command(label="退出",command = self.cancel)
        helpmenu.add_command(label="规则介绍...    F1",command = self.helpEvent1)
        helpmenu.add_command(label="名言...",command = self.helpEvent2)

        self.root.bind('<F2>',self.gameEvent1)
        self.root.bind('<F4>',self.gameEvent2)
        self.root.bind('<F5>',self.gameEvent4)
        self.root.bind('<F1>',self.helpEvent1)
        self.root.protocol("WM_DELETE_WINDOW",self.cancel)
        #导入背景图片
        bgimg = PhotoImage(file = 'bg.gif')
        #创建主界面
        self.frame = GameFrame(self,bgimg)
        #主循环
        self.root.mainloop()
    def gameEvent1(self,event=None):
        #游戏菜单项”新游戏“
        flag = askokcancel('新游戏','你确定要放弃当前游戏开始新游戏么?')
        if flag:
            self.frame.newGame()
    def gameEvent2(self,event=None):
        #游戏菜单项”得分“
        self.frame.showScoreDialog()
    def gameEvent3(self):
        #游戏菜单项”背景音乐“
        if self.v.get() == 0:
            winsound.PlaySound(self.s,winsound.SND_PURGE)
        else:
            self.s = winsound.PlaySound('m1.wav',
                                    winsound.SND_ASYNC+winsound.SND_LOOP)
    def gameEvent4(self,event=None):
        #游戏菜单项”选项“
        optionDialog = OptionDialog(self.root,self.frame.speed / 100 - 1)
        if not optionDialog.isCancel:
            self.frame.speed = 100 + optionDialog.v.get() * 100
    def helpEvent1(self,event=None):
        #帮助菜单项”规则介绍“
        HelpDialog(self.root)
    def helpEvent2(self):
        #帮助菜单项”名言“
        SayDialog(self.root)
    def cancel(self):
        #关闭音乐,退出游戏
        winsound.PlaySound(self.s,winsound.SND_PURGE)
        self.root.destroy()

       
def main():
    Game()

    
if __name__ == '__main__': 
    main()

五、效果展示

1)游戏开始

2)游戏界面

​3)左上角帮助菜单

4)游戏结束排名

到此这篇关于Python Pygame实战之红心大战游戏的实现的文章就介绍到这了,更多相关Python Pygame 红心大战内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Python网络爬虫神器PyQuery的基本使用教程

    Python网络爬虫神器PyQuery的基本使用教程

    这篇文章主要给大家介绍了关于Python网络爬虫神器PyQuery的基本使用教程,文中通过示例代码介绍的非常详细,对大家学习使用PyQuery具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧。
    2018-02-02
  • Python中Numpy mat的使用详解

    Python中Numpy mat的使用详解

    这篇文章主要介绍了Python中Numpy mat的使用详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-05-05
  • 利用Python如何将数据写到CSV文件中

    利用Python如何将数据写到CSV文件中

    在数据分析中经常需要从csv格式的文件中存取数据以及将数据写书到csv文件中。下面这篇文章主要给大家介绍了关于利用Python如何将数据写到CSV文件中的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考下
    2018-06-06
  • zookeeper python接口实例详解

    zookeeper python接口实例详解

    这篇文章主要介绍了zookeeper python接口实例详解,具有一定借鉴价值,需要的朋友可以参考下
    2018-01-01
  • Django使用Channels实现WebSocket的方法

    Django使用Channels实现WebSocket的方法

    WebSocket是一种在单个TCP连接上进行全双工通讯的协议。WebSocket允许服务端主动向客户端推送数据。这篇文章主要介绍了Django使用Channels实现WebSocket,需要的朋友可以参考下
    2019-07-07
  • Python中判断input()输入的数据的类型

    Python中判断input()输入的数据的类型

    在pyhton中,经常会用到input()语句,但是input()语句输入的内容只能是字符串类型,而我们经常要输入int类型的数据等,这个时候就需要用到int()方法给输入的内容强制转换,今天小编给大家介绍下Python中判断input()输入的数据的类型,感兴趣的朋友跟随小编一起看看吧
    2022-11-11
  • 使用python和pygame绘制繁花曲线的方法

    使用python和pygame绘制繁花曲线的方法

    本篇文章主要介绍了使用python和pygame绘制繁花曲线的方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-02-02
  • pycharm进入时每次都是insert模式的解决方式

    pycharm进入时每次都是insert模式的解决方式

    这篇文章主要介绍了pycharm进入时每次都是insert模式的解决方式,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-02-02
  • Python中字典常用操作的示例详解

    Python中字典常用操作的示例详解

    字典是Python必用且常用的数据结构,本文主要为大家梳理了一下常用的字典操作:初始化、合并字典、字典转Pandas等,需要的可以参考一下
    2022-05-05
  • 本地文件上传到七牛云服务器示例(七牛云存储)

    本地文件上传到七牛云服务器示例(七牛云存储)

    这篇文章主要介绍了使用PYTHON把本地文件上传到七牛云服务的方法,开发环境是Python 2.7,大家参考使用吧
    2014-01-01

最新评论