• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            隨筆 - 97, 文章 - 22, 評論 - 81, 引用 - 0
            數據加載中……

            Pygame游戲開發 之四

            Pygame游戲開發之四

            初學乍練

            Pygame中除了Group這個Sprite的容器類,還有一些繼承自Group的容器類,它們分別是RenderUpdatesOrderedUpdates、LayeredUpdates、LayeredDirty。

            pygame.sprite.RenderUpdates繼承自pygame.sprite.Group,它相對于Group的不同點就是重寫了Groupdraw函數,原型是:RenderUpdates.draw(surface) : return Rect_list它將所有它包含的Sprites繪制到surface上,和Group.draw一樣。但是這個函數返回了一系列的屏幕上發生改變的矩形列表,這個矩形列表應該被傳遞到pygame.display.update中去。

            pygame.sprite.OrderedUpdates繼承自pygame.sprite.RenderUpdates,它繪制Sprite的時候是以Sprite被添加時的順序來繪制的。

            pygame.sprite. LayeredUpdates繪制的方式和pygame.sprite.OrderedUpdates類似,只是它引入了圖層的概念。你可以通過'default_layer'設置默認圖層,或者一個整數來表示圖層。如果添加的Sprite自己有一個layer的圖層屬性那么就是用這個,否則在添加Sprite的時候,在pygame.sprite. LayeredUpdates(*sprites, **kwarges)中的**kwarges參數中對layer屬性進行設置,如果兩者都沒有設置,那么系統將采用默認圖層。

            pygame.sprite. LayeredDirty繼承自pygame.sprite.LayeredUpdates,它是DirtySprites的專用容器類,繼承了Group的所有操作。

            因為我們需要用到DirtySprite,所以之前我們用到的所有Group類都替換成LayeredDirty,而且為了便于擴展,我們將LayeredDirty再封裝一層,以便添加新的成員函數。

            class gameManager(pygame.sprite.LayeredDirty) :

                def __init__(self, selfdata) :

                    pygame.sprite.LayeredDirty.__init__(self)

                def keyevent(self, keypressed) :

                    for son in self.sprites() :

                        son.keyevent(keypressed)

                   定義一個gameManager類,它繼承了pygame.sprite.LayeredDirty,并且添加一個keyevent的成員函數,用于對鍵盤事件進行處理,它的操作很簡單,調用它容器里的元素的keyevent函數,它容器里的元素有可能是另一個gameManager、或者是一個單純的RenderObject。如果是gameManager的話就會進行遞歸調用,直到是RenderObject,所以我們還要給RenderObject定義一個keyevent函數,像這樣:

            class RenderObject(pygame.sprite.DirtySprite) :

                   … …

                def keyevent(self, keypressed) :

                    pass

            我們把keyevent定義在基類RenderObject中,并且用pass表示它什么都不做,然后等著子類去實現它。

            class Player(RenderObject) :

                   … …

                def keyevent(self, keypressed) :

                    left_or_right = keypressed[K_RIGHT] - keypressed[K_LEFT]

                    up_or_down    = keypressed[K_DOWN] - keypressed[K_UP]

                    self.move(left_or_right, up_or_down)

            PlayerRenderObject的一個子類,我們可以用以上的語句實現Player的行走,keypressed其實是pygame.key.get_pressed(),用于獲取當前鍵盤的某個鍵是否按下的映射表。

                   接下來我們用之前的模塊來寫一個游戲的背景,利用圖4-1的資源拼接出圖4-2的圖像,并且讓它在屏幕Y軸方向進行滾屏操作。

            4-1

            4-2

                   首先要明確的一點就是圖片的四個方向必須是可拼接的,也就是說用9張相同的圖片拼成一個九宮格,用肉眼是分辨不出它們是九張圖片的,看到的是一個完整的圖像。

                   我們定義一個RenderBack類:

            class RenderBack(RenderObject) :

                def __init__(self, selfdata) :

                    RenderObject.__init__(self, selfdata)

                          do_init()

            def update(self) :

                   RenderObject.update(self)

                          do_update()

            首先確定做這么一個背景需要用到的數據,背景的寬高是肯定要的,而且要求它在Y方向做滾屏操作,所以這個背景的實際的高肯定要比屏幕上顯示高還要長(我們假設他是圖4-1圖片高的整數倍),那么我們用一個四元組(x, y, z, w)來表示需要知道的數據:

            x : 背景圖資源在images[]的索引號

            y : 生成圖的寬

            z : 生成圖的高

            w: 生成圖在豎直方向的原圖圖塊的數量

            這個四元組是作為selfdata被傳到__init__函數中進行初始化的,并且可以在配置文件中修改它的值。

                   因為最后的圖像要進行Y方向的滾屏操作,所以我們還需要定義一個Y方向的偏移量,以便每次繪制的時候進行偏移操作。這樣初始化就可以這么寫:

            self.image       = pygame.Surface( (int(selfdata[1]), int(selfdata[2])) )

            self.rect         = self.image.get_rect()

            self.source_rect   = self.image.get_rect()

                   

            self.backlen      = int(selfdata[3])

            self.backimage    = RenderObject( [ selfdata[0] ] )

            self.top_offset    = self.backimage.image.get_height() * self.backlen - self.rect.height

            這里需要注意的是,LayeredDirty在對DirtySprite進行繪制的時候總是對sprite.image的內容進行繪制的,所以在RenderBack里我們需要額外定義一個backimage來對原圖進行緩存,然后通過循環將它blitimage上。這樣RenderBack的擁有者才能找到image將它繪制出來。

                   對于update函數,每一幀我們需要更新self.top_offset這個Y方向的偏移量,并且根據它更新image圖像。具體實現如下:

            def update(self) :

            RenderObject.update(self)

            xstep = self.backimage.image.get_width()

            ystep = self.backimage.image.get_height()

            self.top_offset -= 3

             

            y_offset = self.top_offset % ystep

            for x in range(0, self.rect.width + xstep, xstep) :

            self.image.blit( self.backimage.image, (x, 0), ((0, y_offset), (xstep, ystep - y_offset) ) )

             

            for y in range(ystep - y_offset, self.rect.height + ystep, ystep) :

            for x in range(0, self.rect.width + xstep, xstep) :

            self.image.blit( self.backimage.image, (x, y), ((0,0), (xstep, ystep) ) )

             

            (xstep, ystep)是原圖的尺寸,每一幀我們對self.top_offset進行自減操作(效果就是讓最后的圖像有一種向上延伸的感覺,也就是滾屏的效果)。

            然后我們分x方向和y方向進行討論,因為x方向沒有滾屏操作,所以只要用現有的圖片填充整個寬度即可,比如原圖是W=100個像素,需要填充的背景是D=250個像素,那么水平方向至少需要用到[ (D + W – 1) / W ] = 3張原圖([x]表示比x小的最大整數,即取下整)。對于豎直方向,因為有滾屏操作,所以有兩部分組成:第一排(從原圖的y方向某個位置粘貼過來的)以及非第一排(總是粘貼原圖)。將self.top_offset ystep 取模,得到的是需要繪制圖像的最上方在原圖的偏移量,通過它可以繪制第一排,然后再調用兩層for循環繪制后面幾排。這樣隨著時間的推移,一個Y方向的滾屏效果就出來了。(未完待續)

             

            posted on 2011-04-28 02:19 英雄哪里出來 閱讀(3324) 評論(0)  編輯 收藏 引用 所屬分類: Pygame

            伊人久久大香线焦AV综合影院| 久久99国产精品尤物| 久久精品无码午夜福利理论片| 午夜天堂精品久久久久| 一本伊大人香蕉久久网手机| 国产成人精品久久亚洲| 欧美黑人激情性久久| 久久综合久久综合亚洲| 久久精品蜜芽亚洲国产AV| 品成人欧美大片久久国产欧美... 品成人欧美大片久久国产欧美 | 久久播电影网| 91久久婷婷国产综合精品青草| 国产精品99久久久久久宅男| 国产69精品久久久久9999APGF| 色综合合久久天天综合绕视看| 麻豆一区二区99久久久久| 久久99国产一区二区三区| 久久婷婷国产综合精品| 伊人久久大香线蕉综合热线| 国产精品激情综合久久| 国产午夜免费高清久久影院| 亚洲国产成人久久综合碰| 国产—久久香蕉国产线看观看| 久久婷婷五月综合国产尤物app | 国产精品伊人久久伊人电影| 欧美牲交A欧牲交aⅴ久久| 亚洲国产综合久久天堂 | 99久久精品免费观看国产| 国产精品中文久久久久久久| 国内精品久久久久影院一蜜桃| 天天综合久久久网| www.久久精品| 热99RE久久精品这里都是精品免费 | 久久精品人人做人人爽97| 久久久久亚洲精品日久生情| 国内精品伊人久久久久网站| 久久免费的精品国产V∧| 久久精品国产99国产精品亚洲 | 国产真实乱对白精彩久久| 日本三级久久网| 国产福利电影一区二区三区久久久久成人精品综合 |