Nauč se Python > Kurzy > Hadí workshop @CodeWeekEU > Had > Logika hry Had

Logika hry

Už umíš vykreslit hada ze seznamu souřadnic. Hadí videohra ale nebude jen „fotka“. Seznam se bude měnit a had se bude hýbat!

Rozhýbejme hada

Tvůj program teď, doufám, vypadá nějak takhle:

from pathlib import Path

import pyglet

TILE_SIZE = 64
TILES_DIRECTORY = Path('snake-tiles')

snake = [(1, 2), (2, 2), (3, 2), (3, 3), (3, 4), (3, 5), (4, 5)]
food = [(2, 0), (5, 1), (1, 4)]

red_image = pyglet.image.load('apple.png')
snake_tiles = {}
for path in TILES_DIRECTORY.glob('*.png'):
    snake_tiles[path.stem] = pyglet.image.load(path)

window = pyglet.window.Window()

@window.event
def on_draw():
    window.clear()
    pyglet.gl.glEnable(pyglet.gl.GL_BLEND)
    pyglet.gl.glBlendFunc(pyglet.gl.GL_SRC_ALPHA, pyglet.gl.GL_ONE_MINUS_SRC_ALPHA)
    for x, y in snake:
        source = 'end'     # (Tady případně je nějaké
        dest = 'end'       #  složitější vybírání políčka)
        snake_tiles[source + '-' + dest].blit(
            x * TILE_SIZE, y * TILE_SIZE, width=TILE_SIZE, height=TILE_SIZE)
    for x, y in food:
        red_image.blit(
            x * TILE_SIZE, y * TILE_SIZE, width=TILE_SIZE, height=TILE_SIZE)

pyglet.app.run()

Zkus těsně nad řádek pyglet.app.run doplnit funkci, která se bude volat každou šestinu vteřiny a přidá hadovi políčko navíc:

def move(dt):
    x, y = snake[-1]
    new_x = x + 1
    new_y = y
    new_head = new_x, new_y
    snake.append(new_head)

pyglet.clock.schedule_interval(move, 1/6)

Funguje? Tak do téhle funkce ještě přidej del snake[0], aby had nerostl donekonečna. Víš co tenhle příkaz dělá? Jestli ne, koukni znovu na poznámky k seznamům!

Zvládneš funkci upravit tak, aby had lezl nahoru?

Jestli ano, gratuluji! Zvývá směr hada ovládat šipkami na klávesnici, a většina hry bude hotová!

Třída pro stav

Než uděláme interaktivního hada, zkusíme trošku uklidit. Program se nám rozrůstá a za chvíli bude složité se v něm vyznat.

Stav hry máme zatím ve dvou seznamech: snake a food. Časem ale bude podobných proměnných víc.

Abychom je měli všechny pohromadě, vytvoříme pro stav třídu.

Na všechno, co se ve hře může stát, nadefinujeme metody. Zatím budou dvě: začátek hry a pohyb hada.

Na začátku hry se zavolá metoda __init__. Má trochu divné jméno se dvěma podtržítkama na každé straně. Podle toho Python ví, že tahle metoda je speciální a se má volat při vytvoření objektu.

Metoda __init__ nastaví celý stav hry jako atributy. Stav hry je všechno, co potřebujeme o hře vědět a může se to časem měnit. V našem případě to zatím budou souřadnice hada a jídla.

Metoda move, kterou budeme volat při každém „tahu“ hry, je bude tyhle atributy měnit.

Pro funkčnost, kterou zatím náš had umí, bude třída se stavem vypadat následovně. Přidej ji do programu hned za nastavení konstant.

class State:
    def __init__(self):
        self.food = [(2, 0), (5, 1), (1, 4)]
        self.snake = [(0, 0), (1, 0)]

    def move(self):
        old_x, old_y = self.snake[-1]
        new_x = old_x + 1
        new_y = old_y
        new_head = new_x, new_y
        self.snake.append(new_head)
        del self.snake[0]

Použij prosím pro třídu jméno State a i atributy pojmenuj podle materiálů (snake, food, a později i další). Bude se ti to hodit.

Všimni si, že metody berou argument self. To označuje konkrétní objekt, stav hry se kterým metoda pracuje nebo který mění. Ke všem atributům přistupují pomocí tečky – self.jméno_atributu.

Tak, máme třídu se stavem. No jo, ale jak ji teď použít?

Na to potřebuješ ještě několik změn:

  • Nastavování seznamů snake a food zruš; místo nich nastav jedinou proměnnou state na nový stav:

    state = State()
    
  • Místo snake a food ve funkci on_draw použij state.snake a state.food – atributy našeho stavu.

    Všimni si že tady nepoužíváme self, což je jméno které používají jen metody v rámci třídy. Jinde musíme pojmenovat konkrétní objekt, se kterým pracujeme.

  • Funkci move přepiš tak, aby jen volala metodu state.move:

    def move(dt):
        state.move()
    

    Všimni si že ani tady nepoužíváme self.

Povedlo se? Funguje to jako předtím? Pro kontrolu můžeš svůj program porovnat s mým (ale nejde o jediné správné řešení):

Řešení

Ovládání

Nyní k onomu slíbenému ovládání. Respektive nejdřív k změnám směru.

Had ze hry se plazí stále stejným směrem, dokud hráč nezmáckne klávesu. Had z naší ukázky se plazí doprava. Jestli jsi to ještě neudělal/a, zkus zařídit, aby se místo toho plazil nahoru.

Řešení

A co dolů?

Řešení

Aby si had „pamatoval“ kam se zrovna plazí, je potřeba mít směr jako součást stavu hry. Uložme ho tedy do atrubutu jménem snake_direction.

Co tam ale přesně uložit? Jak reprezentovat směr v Pythonu – pomocí čísel, n-tic a tak?

Mřížka s X a Y souřadnicemi

Asi nejpříhodnější řešení je uložit si o kolik políček se had má posunout, a to zvlášť v x-ovém a zvlášť v y-ovém směru. Čili jako dvojici:

  • (1, 0) = doprava (o jedno políčko v kladném x-ovém směru; v y-ovém neposouvat)
  • (-1, 0) = doleva (o jedno políčko v záporném x-ovém směru)
  • (0, 1) = nahoru (+y)
  • (0, -1) = dolů (-y)

Nový atribut přidej do metody __init__ ve stavu:

        self.snake_direction = 0, 1

A v metodě move změň nastavování new_x a new_y podle nového atributu:

        dir_x, dir_y = self.snake_direction
        new_x = old_x + dir_x
        new_y = old_y + dir_y

Směr hada teď můžeš měnit změnou snake_direction__init__. Funguje to? (Jestli ne, oprav to – a jestli to nejde, zavolej někoho na pomoc!)

Nyní zbývá atribut snake_direction měnit, když uživatel něco stiskne na klávesnici. To už je doména Pygletu.

Je potřeba přidat funkci, která reaguje na stisk klávesy. Aby Pyglet tuhle funkci našel a uměl zavolat, musí se jmenovat on_key_press, musí mít dekorátor @window.event, a musí brát dva parametry: číslo klávesy, která byla zmáčknutá a informace o modifikátorech jako Shift nebo Ctrl:

@window.event
def on_key_press(key_code, modifier):
    ...

Druhý parametr nebude potřeba, ale musí v hlavičce funkce být.

Podle prvního ale nastav aktuální směr hada. Čísla kláves jsou definována v modulu pyglet.window.key jako konstanty se jmény LEFT, ENTER, Q či AMPERSAND . My použijeme šipky – LEFT, RIGHT, UP a DOWN:

@window.event
def on_key_press(key_code, modifier):
    if key_code == pyglet.window.key.LEFT:
        state.snake_direction = -1, 0
    if key_code == pyglet.window.key.RIGHT:
        state.snake_direction = 1, 0
    if key_code == pyglet.window.key.DOWN:
        state.snake_direction = 0, -1
    if key_code == pyglet.window.key.UP:
        state.snake_direction = 0, 1

Tuhle funkci je potřeba dát někam za nastavení window (aby byl k dispozici window.event) a před pyglet.app.run() (protože nastavovat ovládání až potom, co hra proběhne, je zbytečné). Nejlepší je ji dát vedle jiné funkce s dekorátorem @window.event, aby byly pěkně pohromadě.

Funguje to? Můžeš ovládat směr hada? To je skvělé! Určitě ale při zkoušení narazíš na pár věcí, které je potřeba dodělat:

  • Had by neměl mít možnost vylézt ven z okýnka.
  • Had by měl jíst jídlo a růst.
  • Hra by měla skončit, když had narazí sám do sebe nebo do okraje okna.

Pojďme je vyřešit, jednu po druhé.

Zatím dobrý, teď ale narazíme

„Hadí“ hry jako ta naše mají dvě varianty: buď je kolem hřiště „zeď“ a hráč při nárazu do okraje prohraje, nebo je hřiště „nekonečné“ – had okrajem proleze a objeví se na druhé straně. My naprogramujeme tu první variantu – zeď.

Abys zjistil/a, jestli had „vylezl“ z levého okraje okna ven, je potřeba zkontrolovat, jestli x-ová souřadnice hlavy je menší než 0. To je dobré udělat hned poté, co nové souřadnice hlavy získáš – konkrétně hned za řádkem new_head = new_x, new_y v metodě move.

A co při takovém nárazu udělat? Pro začátek bude nejjednodušší ukončit hru. Na to má Python funkci exit(), která funguje podobně jako když v programu nastane chyba. Jen místo chybového výpisu ukáže daný text.

Ukončení programu není příliš příjemný způsob, jak říct hráčovi že prohrál. Za chvíli ale tuhle část předěláme, tak prozatím tenhle jednoduchý způsob postačí.

    def move(self):
        old_x, old_y = self.snake[-1]
        dir_x, dir_y = self.snake_direction
        new_x = old_x + dir_x
        new_y = old_y + dir_y
        new_head = new_x, new_y

        # Nový kód – kontrola vylezení z hrací plochy
        if new_x < 0:
            exit('GAME OVER')

        self.snake.append(new_head)
        del self.snake[0]

Věřím, že zvládneš udělat stejnou kontrolu pro vylezení ze spodního okraje.

Jak ale ošetřit ty zbylé okraje – pravý a horní? Na to je potřeba znát velikost okýnka. A tu zná Pyglet; třída se stavem by k okýnku neměla mít přístup!

Na velikosti herní plochy závisí chování hry. Tahle informace tedy bude tedy muset být součást stavu. Pro začátek nějakou velikost – třeba 10×10 – nastav v __init__:

        self.width = 10
        self.height = 10

A pak zařiď, aby po nárazu na neviditelnou stěnu kolem hřiště velkého 10×10 políček hra skončila. Pořádně vyzkoušej všechny varianty – severní, jižní, východní i západní zeď. Had je virtuální, nemusíš se bát že mu z toho vyroste boule.

Řešení

A pak v souboru se hrou hned po tom co vytvoříš stav (state = State()) a okýnko (window) nastav opravdovou velikost. Použij celočíselné dělení, aby počet políček byl v celých číslech:

state.width = window.width // TILE_SIZE
state.height = window.height // TILE_SIZE

Krmení

Tak. Had je v kleci, už nemůže vylézt. Co dál?

Teď se musíme o hada postarat: pravidelně ho krmit. Ale ještě předtím je potřeba ho naučit, jak se vůbec jí – na naši potravu ještě není zvyklý. Když to zvládneme, poroste jako z vody!

Konkrétně musíme hlavně zajistit, aby když se had připlazí na políčko s jídlem, tak jídlo zmizelo. K tomu se dá použít:

  • operátor in, který zjišťuje jestli něco (třeba souřadnice) je v nějakém seznamu (třeba seznamu souřadnic jídla), a
  • metoda remove, která ze seznamu odstraní daný prvek (podle hodnoty prvku – na rozdíl od del, který maže podle pozice).

Za kontrolu vylezení z hrací plochy potřebujeme dát kód, který dělá následující:

  • Pokud je nová pozice hlavy v seznamu souřadnic jídla:
    • Odeber tuhle pozici ze seznamu souřadnic jídla

Zvládneš ho napsat?

Řešení

Vyzkoušej, jestli to funguje. Had by měl jíst jídlo.

Ještě ale zbývá zařídit, aby po každém soustu trochu povyrostl. Ale jak? Kterým směrem má růst?

Tady je dobré se podívat na existující kód a uvědomit si, co dělá.

Náš had se plazí tak, že napřed vepředu povyroste (pomocí append) a potom se vzadu zmenší (pomocí del self.snake[0]).

Aby tedy po snězení jídla vyrostl, stačí přeskočit ono zmenšování! Ono přeskočit znamená podmínit, pomocí if. Logika jezení a zmenšování hada tedy bude:

  • Když had sní jídlo, jídlo zmizí. Had se nezmenší.
  • Jinak (tedy když had nesní jídlo) se had zmenší (a tudíž neroste).

Neboli přeloženo do Pythonu:

        if new_head in self.food:
            self.food.remove(new_head)
        else:
            del self.snake[0]

Pro ty, co se začínají ztrácet, dám k dispozici celou metodu move. Běda ale těm, kdo opisují kód aniž mu rozuměli!

Řešení

Nové jídlo

Když už had umí jíst, je potřeba mu zajistit pravidelný přísun jídla. Nejlépe tak, že se každé snězené jídlo nahradí novým.

Přidej do třídy State následující novou metodu, která umí přidat jídlo:

    def add_food(self):
        x = 0
        y = 0
        position = x, y
        self.food.append(position)

Pak tuhle metodu zavolej – najdi v programu kód, který se provádí když je potřeba přidat nové jídlo, a přidej tam následující řádek:

            self.add_food()

Tahle metoda přidává jídlo na pozici (0, 0), tedy stále do stejného rohu. Bylo by ale fajn, kdyby se nové jídlo objevilo vždycky jinde, na náhodném místě. Na to můžeme použít funkci random.randrange, která vrací náhodná celá čísla. Vyzkoušej si ji (z jiného souboru, třeba experiment.py):

import random

print('Na kostce padlo:', random.randrange(6))

Čím se liší random.randrange od klasické hrací kostky? Uměl/a bys program upravit tak, aby padalo 1 až 6?

Je tahle změna užitečná pro naši hru? Jaký rozsah čísel potřebujeme pro hadí jídlo?

Až na to přijdeš, zkus přidat náhodu do programu: jídlo by se mělo objevit na úplně náhjodném políčku na herní ploše.

Nezapomeň na import random – to patří na úplný začátek souboru. Další změny ale už dělej jen v metodě add_food.

Řešení

Až to budeš testovat, asi zjistíš, že úplně náhodné políčko není ideální. Občas se totiž jídlo objeví na políčku s hadem, nebo dokonce na jiném jídle. Je proto dobré tuhle situaci zkontrolovat, a když volba padne na plné políčko, jídlo nepřidávat:

        if (position not in self.snake) and (position not in self.food):
            self.food.append(position)

Když ale zkusíš tohle, zjistíš, že občas se nové jídlo vůbec nepřidá. To taky není vhodná varianta – had by tak měl hlad. Co s tím?

Překvapivě dobré (i když ne úplně ideální) řešení je zkusit políčko vybrat několikrát. Když padne prázdné políčko, šup tam s jídlem; když padne plné, tak to prostě zkusit znovu.

Je ale potřeba počet pokusů omezit, aby v situaci, kdy je pole úplně plné, počítač nevybíral donekonečna. Řekněme že když se na 100 pokusů nepodaří prázdné políčko vybrat, vzdáme to.

Metoda add_food po všech úpravách bude vypadat takhle:

    def add_food(self):
        for try_number in range(100):
            x = random.randrange(self.width)
            y = random.randrange(self.height)
            position = x, y
            if (position not in self.snake) and (position not in self.food):
                self.food.append(position)
                # Ukončení funkce ("vyskočí" i z cyklu for)
                return

Jestli ti to funguje, ještě zařiď, aby na začátku hry bylo jídlo na náhodných pozicích.

Řešení

Konec

Had teď může narůst do obrovských rozměrů – a lze prohrát jen tím, že narazí do stěny. Zaříďme teď, aby hra skončila i když narazí sám do sebe.

Jak na to? Do metody move, vedle kontrola vylezení z hrací plochy, dej kód který udělá následující:

  • Pokud jsou souřadnice nové hlavy už součást hada:
    • Ukonči hru (podobně jako po nárazu do stěny).

Dokážeš to převést do Pythonu?

Řešení

Hotovo!

Pauza

Není ale dobré při konci hry ukončit celý program a zavřít okýnko.

Lepší je hru „zapauzovat“ a ukázat hráči situaci, do které nešťastného hada dostal, aby se z ní mohl pro příště poučit.

Aby to bylo možné, dáme do stavu hry další atribut: snake_alive. Ten bude nastavený na True, dokud bude had žít. Když had narazí, nastaví se na False, a od té doby se už nebude pohybovat. Je dobré i graficky ukázat, že hadovi není dobře – hráč pak spíš bude zpytovat svědomí.

Zkus zapřemýšlet, kam v kódu patří následující kousky kódu, které prohru implementují:

        # Prvotní nastavení atributu
        self.snake_alive = True
        # Zastavení hada
        self.snake_alive = False
        # Zabránění pohybu
        if not self.snake_alive:
            return
        # Grafická indikace
        if dest == 'end' and not state.snake_alive:
            dest = 'dead'

Řešení

Vylepšení ovládání

Poslední úprava kódu!

Možná si všimneš, zvlášť jestli jsi už nějakou verzi hada hrál/a, že ovládání tvé nové hry je trošku frustrující. A možná není úplně jednoduché přijít na to, proč.

Můžou za to (hlavně) dva důvody.

První problém: když zmáčkneš dvě šipky rychle za sebou, v dalším „tahu“ hada se projeví jen ta druhá. Z pohledu programu to chování dává smysl – po stisknutí šipky se uloží její směr, a při „tahu“ hada se použije poslední uložený směr. S tímhle chováním je ale složité hada rychle otáčet: hráč si musí pohlídat, aby pro každý „tah“ hada nezmáčkl víc než jednu šipku. Lepší by bylo, kdyby se ukládaly všechny stisknuté klávesy, a had by v každém tahu reagoval maximálně jednu. Další by si „schoval“ na další tahy.

Takovou „frontu“ stisků kláves lze uchovávat v seznamu. Přidej si na to do stavu hry seznam (v metodě __init__):

        self.queued_directions = []

Tuhle frontu plň po každém stisku klávesy, metodou append. Je potřeba změnit většinu funkce on_key_press – místo změny atributu se nový směr přidá do seznamu. Abys nemusel/a psát čtyřikrát append, můžeš uložit nový směr do pomocné proměnné:

@window.event
def on_key_press(key_code, modifier):
    if key_code == pyglet.window.key.LEFT:
        new_direction = -1, 0
    if key_code == pyglet.window.key.RIGHT:
        new_direction = 1, 0
    if key_code == pyglet.window.key.DOWN:
        new_direction = 0, -1
    if key_code == pyglet.window.key.UP:
        new_direction = 0, 1
    state.queued_directions.append(new_direction)

A zpátky k logice. V metodě move místo dir_x, dir_y = self.snake_direction z fronty vyber první nepoužitý prvek. Nezapomeň ho pak z fronty smazat, ať se dostane i na další:

        if self.queued_directions:
            new_direction = self.queued_directions[0]
            del self.queued_directions[0]
            self.snake_direction = new_direction

Zkontroluj, že to funguje.

Zpátky ni krok

Druhý problém s ovládáním: když se had plazí doleva a hráč zmáčkne šipku doprava, had se otočí a hlavou si narazí do krku. Z pohledu programu to opět dává smysl: políčko napravo od hlavy je plné, had na něj tedy nemůže vstoupit a hráč prohrává. Z pohledu hry (a biologie!) ale narážení do krku moc smyslu nedává. Lepší by bylo obrácení směru úplně ignorovat.

A jak poznat opačný směr? Když se had plazí doprava, (1, 0), tak je opačný směr doleva, (-1, 0). Když se plazí dolů, (0, -1), tak naopak je nahoru, (0, 1). Obecně, k (x, y) je opačný směr (-x, -y).

Zatím ale pracujeme s celými n-ticemi, je potřeba obě na x a y „rozbalit“. Kód tedy bude vypadat takto:

            old_x, old_y = self.snake_direction
            new_x, new_y = new_direction
            if (old_x, old_y) != (-new_x, -new_y):
                self.snake_direction = new_direction

Dej ho místo puvodního self.snake_direction = new_direction.

A to je vše?

Gratuluji, máš funkční a hratelnou hru! Doufám že jsi na sebe hrdý/á!

Dej si něco sladkého, zasloužíš si to.


Tady je moje řešení. To se touhle dobou od toho tvého může dost lišit – to je úplně normální. (A nedívej se sem dokud hada nenaprogramuješ sám/sama, Chybami a neustálým zkoušením se člověk učí – a programátor zvlášť. Čtením už vyřešeného se učí hůř.)

Řešení

Najdeš ještě nějaké další vylepšení, které by se dalo udělat?

Zkus třeba následující rozšíření:

  • Každých 30 vteřin hry přibude samo od sebe nové jídlo, takže jich pak bude na hrací ploše víc.

  • Hra se bude postupně zrychlovat.
    (Na to je nejlepší předělat funkci move, aby sama naplánovala, kdy se má příště zavolat. Volání schedule_interval tak už nebude potřeba.)

  • Hadi budou dva; druhý se ovládá klávesami W A S D.
    (Na to je nejlepší udělat novou třídu, Snake, a všechen stav hada přesunout ze State do ní. Ve State pak měj seznam hadů. Téhle změně je potřeba přizpůsobit celý zytek programu.)


Toto je stránka lekce z kurzu, který probíhá nebo proběhl naživo s instruktorem.