`

俄罗斯方块游戏

阅读更多
用python+PYQT4写的一个俄罗斯方块游戏

#!/usr/bin/python
# tetris.py
import sys
import random
from PyQt4 import QtCore, QtGui
class Tetris(QtGui.QMainWindow):
    def __init__(self):
        QtGui.QMainWindow.__init__(self)
        self.setGeometry(300, 300, 180, 380)
        self.setWindowTitle('Tetris')
        self.tetrisboard = Board(self)
        self.setCentralWidget(self.tetrisboard)
        self.statusbar = self.statusBar()
        self.connect(self.tetrisboard, QtCore.SIGNAL("messageToStatusbar(QString)"),
            self.statusbar, QtCore.SLOT("showMessage(QString)"))
        self.tetrisboard.start()
        self.center()
    def center(self):
        screen = QtGui.QDesktopWidget().screenGeometry()
        size =  self.geometry()
        self.move((screen.width()-size.width())/2,
            (screen.height()-size.height())/2)
class Board(QtGui.QFrame):
    BoardWidth = 10
    BoardHeight = 22
    Speed = 300
    def __init__(self, parent):
        QtGui.QFrame.__init__(self, parent)
        self.timer = QtCore.QBasicTimer()
        self.isWaitingAfterLine = False
        self.curPiece = Shape()
        self.nextPiece = Shape()
        self.curX = 0
        self.curY = 0
        self.numLinesRemoved = 0
        self.board = []
        self.setFocusPolicy(QtCore.Qt.StrongFocus)
        self.isStarted = False
        self.isPaused = False
        self.clearBoard()
        self.nextPiece.setRandomShape()
    def shapeAt(self, x, y):
        return self.board[(y * Board.BoardWidth) + x]
    def setShapeAt(self, x, y, shape):
        self.board[(y * Board.BoardWidth) + x] = shape
    def squareWidth(self):
        return self.contentsRect().width() / Board.BoardWidth
    def squareHeight(self):
        return self.contentsRect().height() / Board.BoardHeight
    def start(self):
        if self.isPaused:
            return
        self.isStarted = True
        self.isWaitingAfterLine = False
        self.numLinesRemoved = 0
        self.clearBoard()
        self.emit(QtCore.SIGNAL("messageToStatusbar(QString)"),
            str(self.numLinesRemoved))
        self.newPiece()
        self.timer.start(Board.Speed, self)
    def pause(self):
        if not self.isStarted:
            return
        self.isPaused = not self.isPaused
        if self.isPaused:
            self.timer.stop()
            self.emit(QtCore.SIGNAL("messageToStatusbar(QString)"), "paused")
        else:
            self.timer.start(Board.Speed, self)
            self.emit(QtCore.SIGNAL("messageToStatusbar(QString)"),
                str(self.numLinesRemoved))
        self.update()
    def paintEvent(self, event):
        painter = QtGui.QPainter(self)
        rect = self.contentsRect()
        boardTop = rect.bottom() - Board.BoardHeight * self.squareHeight()
        for i in range(Board.BoardHeight):
            for j in range(Board.BoardWidth):
                shape = self.shapeAt(j, Board.BoardHeight - i - 1)
                if shape != Tetrominoes.NoShape:
                    self.drawSquare(painter,
                        rect.left() + j * self.squareWidth(),
                        boardTop + i * self.squareHeight(), shape)
        if self.curPiece.shape() != Tetrominoes.NoShape:
            for i in range(4):
                x = self.curX + self.curPiece.x(i)
                y = self.curY - self.curPiece.y(i)
                self.drawSquare(painter, rect.left() + x * self.squareWidth(),
                    boardTop + (Board.BoardHeight - y - 1) * self.squareHeight(),
                    self.curPiece.shape())
    def keyPressEvent(self, event):
        if not self.isStarted or self.curPiece.shape() == Tetrominoes.NoShape:
            QtGui.QWidget.keyPressEvent(self, event)
            return
        key = event.key()
        if key == QtCore.Qt.Key_P:
            self.pause()
            return
        if self.isPaused:
            return
        elif key == QtCore.Qt.Key_Left:
            self.tryMove(self.curPiece, self.curX - 1, self.curY)
        elif key == QtCore.Qt.Key_Right:
            self.tryMove(self.curPiece, self.curX + 1, self.curY)
        elif key == QtCore.Qt.Key_Down:
            self.tryMove(self.curPiece.rotatedRight(), self.curX, self.curY)
        elif key == QtCore.Qt.Key_Up:
            self.tryMove(self.curPiece.rotatedLeft(), self.curX, self.curY)
        elif key == QtCore.Qt.Key_Space:
            self.dropDown()
        elif key == QtCore.Qt.Key_D:
            self.oneLineDown()
        else:
            QtGui.QWidget.keyPressEvent(self, event)
    def timerEvent(self, event):
        if event.timerId() == self.timer.timerId():
            if self.isWaitingAfterLine:
                self.isWaitingAfterLine = False
                self.newPiece()
            else:
                self.oneLineDown()
        else:
            QtGui.QFrame.timerEvent(self, event)
    def clearBoard(self):
        for i in range(Board.BoardHeight * Board.BoardWidth):
            self.board.append(Tetrominoes.NoShape)
    def dropDown(self):
        newY = self.curY
        while newY > 0:
            if not self.tryMove(self.curPiece, self.curX, newY - 1):
                break
            newY -= 1
        self.pieceDropped()
    def oneLineDown(self):
        if not self.tryMove(self.curPiece, self.curX, self.curY - 1):
            self.pieceDropped()
    def pieceDropped(self):
        for i in range(4):
            x = self.curX + self.curPiece.x(i)
            y = self.curY - self.curPiece.y(i)
            self.setShapeAt(x, y, self.curPiece.shape())
        self.removeFullLines()
        if not self.isWaitingAfterLine:
            self.newPiece()
    def removeFullLines(self):
        numFullLines = 0
        rowsToRemove = []
        for i in range(Board.BoardHeight):
            n = 0
            for j in range(Board.BoardWidth):
                if not self.shapeAt(j, i) == Tetrominoes.NoShape:
                    n = n + 1
            if n == 10:
                rowsToRemove.append(i)
        rowsToRemove.reverse()
        for m in rowsToRemove:
            for k in range(m, Board.BoardHeight):
                for l in range(Board.BoardWidth):
                    self.setShapeAt(l, k, self.shapeAt(l, k + 1))
        numFullLines = numFullLines + len(rowsToRemove)
        if numFullLines > 0:
            self.numLinesRemoved = self.numLinesRemoved + numFullLines
            self.emit(QtCore.SIGNAL("messageToStatusbar(QString)"),
                str(self.numLinesRemoved))
            self.isWaitingAfterLine = True
            self.curPiece.setShape(Tetrominoes.NoShape)
            self.update()
    def newPiece(self):
        self.curPiece = self.nextPiece
        self.nextPiece.setRandomShape()
        self.curX = Board.BoardWidth / 2 + 1
        self.curY = Board.BoardHeight - 1 + self.curPiece.minY()
        if not self.tryMove(self.curPiece, self.curX, self.curY):
            self.curPiece.setShape(Tetrominoes.NoShape)
            self.timer.stop()
            self.isStarted = False
            self.emit(QtCore.SIGNAL("messageToStatusbar(QString)"), "Game over")
    def tryMove(self, newPiece, newX, newY):
        for i in range(4):
            x = newX + newPiece.x(i)
            y = newY - newPiece.y(i)
            if x < 0 or x >= Board.BoardWidth or y < 0 or y >= Board.BoardHeight:
                return False
            if self.shapeAt(x, y) != Tetrominoes.NoShape:
                return False
        self.curPiece = newPiece
        self.curX = newX
        self.curY = newY
        self.update()
        return True
    def drawSquare(self, painter, x, y, shape):
        colorTable = [0x000000, 0xCC6666, 0x66CC66, 0x6666CC,
                      0xCCCC66, 0xCC66CC, 0x66CCCC, 0xDAAA00]
        color = QtGui.QColor(colorTable[shape])
        painter.fillRect(x + 1, y + 1, self.squareWidth() - 2,
            self.squareHeight() - 2, color)
        painter.setPen(color.light())
        painter.drawLine(x, y + self.squareHeight() - 1, x, y)
        painter.drawLine(x, y, x + self.squareWidth() - 1, y)
        painter.setPen(color.dark())
        painter.drawLine(x + 1, y + self.squareHeight() - 1,
            x + self.squareWidth() - 1, y + self.squareHeight() - 1)
        painter.drawLine(x + self.squareWidth() - 1,
            y + self.squareHeight() - 1, x + self.squareWidth() - 1, y + 1)
class Tetrominoes(object):
    NoShape = 0
    ZShape = 1
    SShape = 2
    LineShape = 3
    TShape = 4
    SquareShape = 5
    LShape = 6
    MirroredLShape = 7
class Shape(object):
    coordsTable = (
        ((0, 0),     (0, 0),     (0, 0),     (0, 0)),
        ((0, -1),    (0, 0),     (-1, 0),    (-1, 1)),
        ((0, -1),    (0, 0),     (1, 0),     (1, 1)),
        ((0, -1),    (0, 0),     (0, 1),     (0, 2)),
        ((-1, 0),    (0, 0),     (1, 0),     (0, 1)),
        ((0, 0),     (1, 0),     (0, 1),     (1, 1)),
        ((-1, -1),   (0, -1),    (0, 0),     (0, 1)),
        ((1, -1),    (0, -1),    (0, 0),     (0, 1))
    )
    def __init__(self):
        self.coords = [[0,0] for i in range(4)]
        self.pieceShape = Tetrominoes.NoShape
        self.setShape(Tetrominoes.NoShape)
    def shape(self):
        return self.pieceShape
    def setShape(self, shape):
        table = Shape.coordsTable[shape]
        for i in range(4):
            for j in range(2):
                self.coords[i][j] = table[i][j]
        self.pieceShape = shape
    def setRandomShape(self):
        self.setShape(random.randint(1, 7))
    def x(self, index):
        return self.coords[index][0]
    def y(self, index):
        return self.coords[index][1]
    def setX(self, index, x):
        self.coords[index][0] = x
    def setY(self, index, y):
        self.coords[index][1] = y
    def minX(self):
        m = self.coords[0][0]
        for i in range(4):
            m = min(m, self.coords[i][0])
        return m
    def maxX(self):
        m = self.coords[0][0]
        for i in range(4):
            m = max(m, self.coords[i][0])
        return m
    def minY(self):
        m = self.coords[0][1]
        for i in range(4):
            m = min(m, self.coords[i][1])
        return m
    def maxY(self):
        m = self.coords[0][1]
        for i in range(4):
            m = max(m, self.coords[i][1])
        return m
    def rotatedLeft(self):
        if self.pieceShape == Tetrominoes.SquareShape:
            return self
        result = Shape()
        result.pieceShape = self.pieceShape
        for i in range(4):
            result.setX(i, self.y(i))
            result.setY(i, -self.x(i))
        return result
    def rotatedRight(self):
        if self.pieceShape == Tetrominoes.SquareShape:
            return self
        result = Shape()
        result.pieceShape = self.pieceShape
        for i in range(4):
            result.setX(i, -self.y(i))
            result.setY(i, self.x(i))
        return result
app = QtGui.QApplication(sys.argv)
tetris = Tetris()
tetris.show()
sys.exit(app.exec_())
分享到:
评论
2 楼 lampeter123 2009-06-20  
运行环境是python 2.6 + pyqt4

pyqt4在以下网址下载:
http://www.riverbankcomputing.co.uk/software/pyqt/download

安装以上软件之后,用python shell 直接运行就可以
1 楼 kungstriving 2009-06-19  
收藏了,正准备找一个这样的源码看看呢
楼主能不能把具体的运行方式再贴出来,需要什么库之类的
谢谢!

相关推荐

    俄罗斯方块游戏-JAVA实现(含双人联机对战).zip

    俄罗斯方块游戏--JAVA实现(含双人联机对战).zip 俄罗斯方块游戏--JAVA实现(含双人联机对战).zip 俄罗斯方块游戏--JAVA实现(含双人联机对战).zip 俄罗斯方块游戏--JAVA实现(含双人联机对战).zip 俄罗斯方块...

    Delphi编写的俄罗斯方块游戏源码

    《Delphi实现的俄罗斯方块游戏源码解析》 Delphi是一种基于Object Pascal语言的集成开发环境,以其高效、简洁的编程风格深受程序员喜爱。在游戏开发领域,尽管不如Unity或Unreal Engine等大型引擎常见,但Delphi...

    俄罗斯方块游戏设计(C++课程设计报告)

    《俄罗斯方块游戏设计——基于C++的MFC实现》 在计算机科学的学习过程中,课程设计是提升实践技能的重要环节。本次我们关注的是一个经典的案例——利用C++和MFC框架设计实现“俄罗斯方块”游戏。这个项目不仅涵盖了...

    俄罗斯方块游戏源代码俄罗斯方块游戏源代码俄罗斯方块游戏源代码C++

    俄罗斯方块游戏源代码俄罗斯方块游戏源代码俄罗斯方块游戏源代码俄罗斯方块游戏源代码俄罗斯方块游戏源代码C++C++C++

    基于MFC的俄罗斯方块游戏

    **基于MFC的俄罗斯方块游戏** MFC(Microsoft Foundation Classes)是微软提供的一套C++库,用于构建Windows应用程序。它封装了Windows API,使得开发者可以使用面向对象的方式来开发Windows程序,包括桌面应用、...

    基于c51单片机的俄罗斯方块游戏.rar

    《基于C51单片机实现的俄罗斯方块游戏详解》 在电子技术与嵌入式系统领域,C51单片机是一个广泛使用的微控制器,因其强大的处理能力和丰富的资源而备受青睐。本项目——“基于C51单片机的俄罗斯方块游戏”,将深入...

    俄罗斯方块游戏源码-Java

    【经典再现,Java俄罗斯方块游戏源码】 踏上编码之旅,揭开经典游戏的神秘面纱!我们的Java版俄罗斯方块源码项目是您深入理解游戏开发、算法设计和面向对象编程的完美起点! 核心特色与优势 跨平台兼容:基于强大...

    C#俄罗斯方块游戏源代码,带背景音乐和音效

    《C#实现的俄罗斯方块游戏源代码解析》 在编程世界中,游戏开发一直是一项深受程序员喜爱的挑战。本文将深入探讨一个基于C#语言的俄罗斯方块游戏源代码,该代码集成了背景音乐和音效,为初学者提供了一个良好的实践...

    俄罗斯方块游戏代码解析

    《VC++实现的俄罗斯方块游戏代码解析》 在编程世界中,俄罗斯方块是一款经典的电子游戏,它的简单规则和无限挑战性使得它成为初学者学习游戏编程的绝佳选择。本资源提供了一款使用VC++编写的俄罗斯方块游戏源代码,...

    开题报告-基于JAVA的俄罗斯方块游戏设计与实现.doc

    【基于JAVA的俄罗斯方块游戏设计与实现】 本课题旨在设计和实现一款基于JAVA的俄罗斯方块游戏。在选择JAVA作为开发语言时,主要考虑了JAVA的多项优势,包括平台无关性、面向对象特性、安全稳定性、多线程支持以及其...

    【文献综述】基于JAVA的俄罗斯方块游戏设计与实现.pdf

    【文献综述】基于JAVA的俄罗斯方块游戏设计与实现 在计算机科学与技术领域,设计和实现基于JAVA的俄罗斯方块游戏是一个受欢迎的实践项目,因为它不仅深受玩家喜爱,而且具有教学价值。俄罗斯方块游戏以其简洁的游戏...

    完整纯C语言俄罗斯方块游戏-----内部含有音频播放

    《C语言实现俄罗斯方块游戏及音频播放》 在编程世界中,俄罗斯方块是一款经典的休闲游戏,它的实现原理和代码结构对初学者来说具有很高的学习价值。本项目以C语言为开发工具,展示了如何从零开始构建一个完整的...

    用C#编写的俄罗斯方块游戏

    《C#实现的俄罗斯方块游戏详解》 俄罗斯方块是一款经典的电子游戏,自1984年诞生以来,就以其独特的玩法和无尽的挑战性吸引了全球无数玩家。本项目是利用C#编程语言在Visual Studio 2008(VS2008)平台上实现的...

    基于嵌入式的俄罗斯方块游戏

    【基于嵌入式的俄罗斯方块游戏】是一款在电视游戏机和掌上游戏机上深受喜爱的经典游戏,其在游戏历史上有着重要地位。本项目利用了ARM嵌入式硬件平台,结合Windows CE 5.0操作系统,对俄罗斯方块进行了设计与实现。 ...

    一个n年前用vb写的俄罗斯方块游戏源代码

    【标题】中的“一个n年前用vb写的俄罗斯方块游戏源代码”表明这是一个基于Visual Basic(VB)编程语言开发的俄罗斯方块游戏的源代码。VB是微软公司开发的一种面向对象的编程语言,尤其在20世纪90年代至21世纪初非常...

    基于stm32的俄罗斯方块游戏

    在本项目"基于STM32的俄罗斯方块游戏"中,开发者利用STM32的处理能力和丰富的外设资源,实现了一个经典的电子游戏——俄罗斯方块。以下将详细介绍该项目涉及的主要知识点: 1. **STM32硬件平台**:STM32系列具有...

    C语言项目开发_-_俄罗斯方块游戏

    ### C语言项目开发:俄罗斯方块游戏 #### 深入理解C语言项目开发实践——以俄罗斯方块游戏为例 俄罗斯方块游戏自其诞生以来,不仅在全球范围内引起了巨大的轰动,更是对游戏产业产生了深远的影响。这款由苏联游戏...

    俄罗斯方块游戏java版

    《简单的俄罗斯方块游戏》是基于Java编程语言开发的一款经典电子游戏。俄罗斯方块自1984年诞生以来,以其简洁的操作和无尽的挑战性吸引了全球无数玩家。在这个Java版的游戏中,开发者旨在重现这个经典游戏的核心玩法...

    俄罗斯方块游戏_俄罗斯方块_

    【俄罗斯方块游戏】是一种经典的电子游戏,由苏联软件工程师阿列克谢·帕基特诺夫于1984年发明。这个游戏以其简洁的设计、易于理解和高度上瘾性著称,吸引了全球无数玩家的喜爱。在本文中,我们将探讨如何使用keil...

    C语言俄罗斯方块游戏源代码

    根据提供的文件信息,本文将详细解析“C语言俄罗斯方块游戏源代码”的相关知识点,包括游戏的基本概念、C语言在游戏开发中的应用以及如何利用C语言实现一个简易版的俄罗斯方块游戏。 ### 一、俄罗斯方块游戏概述 ...

Global site tag (gtag.js) - Google Analytics