• <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的容器類,它們分別是RenderUpdatesOrderedUpdatesLayeredUpdatesLayeredDirty

            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 英雄哪里出來 閱讀(3315) 評論(0)  編輯 收藏 引用 所屬分類: Pygame

            狠狠色丁香久久综合五月| 亚洲熟妇无码另类久久久| 91久久精一区二区三区大全| 精品久久久久久中文字幕| 免费精品99久久国产综合精品| 久久无码一区二区三区少妇| 亚洲人成电影网站久久| 狠狠色丁香婷婷综合久久来 | 久久精品亚洲男人的天堂| 久久久久综合中文字幕| 69久久夜色精品国产69| 青青热久久国产久精品| 伊人久久精品线影院| 精品国产乱码久久久久久呢 | 无码人妻久久一区二区三区免费 | 日本三级久久网| 欧美一区二区三区久久综 | 99久久精品免费看国产一区二区三区 | 亚洲精品蜜桃久久久久久| 国内精品久久久久久久久电影网| 精产国品久久一二三产区区别| 国产精品xxxx国产喷水亚洲国产精品无码久久一区 | 久久精品国产99久久丝袜| 波多野结衣AV无码久久一区| 国产精品日韩深夜福利久久| 久久AV无码精品人妻糸列| 久久人人爽人人精品视频| 91精品国产高清久久久久久国产嫩草 | 久久se精品一区二区影院 | 色播久久人人爽人人爽人人片AV| 99久久国语露脸精品国产| 久久综合给合久久狠狠狠97色69| 国产精品99久久久精品无码| 狠狠久久综合| 久久久久婷婷| 亚洲伊人久久综合中文成人网| 亚洲人成网站999久久久综合 | 久久久WWW成人| 久久亚洲国产精品五月天婷| 精品综合久久久久久88小说| 亚洲七七久久精品中文国产|