Gymnázium, Praha 6, Arabská 16 Předmět Programování, vyučující Tomáš Obdržálek
Hledání Min Ročníkový projekt Filip Mašát, 1.E květen 2014
anotace: Hledání min je logická hra pro jednoho hráče, ve které jde o odhalení všech políček, až na ty kde se nachází miny. Pokud hráč odhalí políčko s minou, hra končí a on prohrál. Hráčovi pomáhají čísla na různých políčkách, které udávají, kolika min se dotýká. Hra je zpracovaná graficky a tento projekt byl vytvořen ve vývojovém prostředí Netbeans za užití programovacího jazyka Java Minesweeper is a single player logic game, the objective of the game is to uncover a minefield without detonating any mines. If the player uncovers a box with a mine, he loses. The numbers in boxes help the player, because they tell the player, how many mines they’re touching. The game is graphically processed and this project was written in Java using Netbeans.
Prohlášení Prohlašuji, že jsem jediným autorem tohoto projektu, všechny citace jsou řádně označené a všechna použitá literatura a další zdroje jsou v práci uvedené. Tímto dle zákona 121/2000 Sb. (tzv. Autorský zákon) ve znění pozdějších předpisů uděluji bezúplatně škole Gymnázium, Praha 6, Arabská 14 oprávnění k výkonu práva na rozmnožování díla (§ 13) a práva na sdělování díla veřejnosti (§ 18) na dobu časově neomezenou a bez omezení územního rozsahu.
Stránka | 1
Zadání projektu Filip Mašát 1.E Funkce Java Hrací pole Možnost označení políčka Každé políčko kde není mina, bude ukazovat, kolika min se dotýká Náhodné naplňování políček minami (vždy stejný počet) Ukončení aplikace v případě kliknutí Bonusové funkce Měření času Pokud hráč klikne na prázdné pole, odhalí se všechna okolní nevyplněná
Stránka | 2
Obsah Hledání min ......................................................................................................................................... 4 Úvod .................................................................................................................................................... 4 1 Instalace............................................................................................................................................ 4 2 Vytváření mé hry .............................................................................................................................. 4 2.1 Vytvoření hracího pole .............................................................................................................. 4 2.2 Naplňování pole......................................................................................................................... 5 2.3 Ovládání myší ............................................................................................................................ 5 2.4 Odhalování polí.......................................................................................................................... 5 3 Problémy a zádrhely ......................................................................................................................... 6 3.1 ArrayOutOfBoundsException .................................................................................................... 6 3.2 Označováni a od-označování políček......................................................................................... 6 3.3 Zobrazování nul ......................................................................................................................... 6 3.4 Počet min................................................................................................................................... 7 Závěr .................................................................................................................................................... 8 Zdroje .................................................................................................................................................. 8
Stránka | 3
Hledání min Úvod Jako ročníkový projekt jsem si vybral hru hledání min. Zvolil jsem si ji z toho důvodu, že to byla první hra, která mě napadla, kterou bych mohl zvládnout. Hledání min je hra, která se poprvé objevila v roce 1990 jako součást Windows Entertainment Pack.
1 Instalace Otevřeme si složku Hledání min a spustíme soubor HledaniMin.jar. Instalace není potřeba.
Obrázek 1- Historie vzhledu hledání min
2 Vytváření mé hry 2.1 Vytvoření hracího pole Celý projekt jsem dělal v NetBeansech v MyPanelu, který jsme obdrželi od pana profesora Obdržálka. Je to třída, která má v sobě již načrtnuté metody jako paintComponent, MouseListener, KeyListener a samozřejmě Main. Jako jednu z prvních věcí jsem si vytvořil třídu Tlačítko a v ní objekty typu int: mina a dotyky. Postupem času jsem k nim přidal objekty typu boolean: wasClicked a oznaceni (viz. Obrázek 2). Objekt mina jsem měl původně typu boolean, avšak potom jsem zkusil omezit počet min pomocí random.nextInt(). To je však neomezilo úplně, ale na chvíli to účel splnilo. Int dotyky určuje, počet min kterých se dané políčko dotýká, boolean wasClicked říká, zda hráč již na tlačítko klikl a boolean oznaceni nám určuje, zda hráč toto dané políčko označil z důvodu, že si myslí, že je zde mina. Nejdříve jsem si vytvořil dvojrozměrné pole, které jsem naplňoval jednotlivými objekty. Poté jsem ho v paintComponentu vytiskl. Na vytisknuté pole jsem potom položil modré čtverce, které zakrývají jednotlivá políčka.
Obrázek 2- Objekty v třídě Tlacitko
Stránka | 4
2.2 Naplňování pole Pole jsem naplňoval v metodě Main. Pole jsem nejdříve pouze vytvořil a naplnil. Všechny objekty, které nyní byly v poli, byly stejné. Poté jsem vytvořil cyklus, který se 15x opakoval a během každého opakování si náhodně vygeneroval dvě čísla, tedy souřadnice kde leží mina. Poté následoval cyklus, kde jsem se ptal, kolika min se dané tlačítko dotýká. Kvůli tomuto jsem musel udělat obrovské množství podmínek, protože pokud bych ho neudělal, tak by se objevovala chyba: ArrayOutOfBoundsException. V překladu: ptal jsem se na pozice, které neexistují. Objekty wasClicked a oznaceni se neměnili. 2.3 Ovládání myší Tato část mi dala velice zabrat. Před tím jsem s metodou mouseListener nikdy nepracoval a bylo pro mě obtížné pochopit její podstatu. Zde se začali měnit objekty wasClicked a oznaceni. V mouseListener jsem si musel vytvořit podmínku, která říká, na které políčko jsem klikl (viz. Obrázek 3). Dále jsem vytvořil podmínku, díky které program ví, zda je již konec. Pokud konec ještě nebyl, tak se u daného tlačítka změní wasClicked na true. Zde jsem také musel naprogramovat označování jednotlivých políček. Udělal jsem to stejně jakou u odhalování políček, jen s tím rozdílem, že to je pro pravé tlačítko myši a na true se mění oznaceni. Avšak to nebylo vše, protože jsem musel nějak zařídit, abych dané označení mohl vrátit zpět. K tomuto účelu jsem použil prostřední tlačítko myši, které vypadalo v podstatě stejně jako pravé, jen s tím rozdílem, že se zde oznaceni menilo na false. Obrázek 3- Podmínka pro zjištění pozice v poli
2.4 Odhalování polí Jak jsem říkal v předchozí části, vždy pokud kliknete levým tlačítkem myši na políčko tak se změní wasClicked na true. To v paintComponent tudíž musí způsobit něco, díky čemuž se čtverec na dané pozici nevykreslí. K tomuto jsem použil podmínku if (!miny[i][j].wasClicked). V této podmínce jsem začal kreslit čtverce, tudíž pokud se wasClicked rovná true, tento krok se přeskočí a tím pádem se čtverec nevykresli, respektivě nevyplní. Stejně tak funguje i označování políček, pouze v podmínce je oznaceni a čtverec se vyplní žlutou barvou. HledaniMin
Tlacitko
Objekty: mina, dotyk wasClicked, oznaceni
main
MyPanel
BufferedImage
MyPanel
paintComponent
mouseListener
mouseReleased
Obrázek 4-Diagram struktury projektu
Stránka | 5
3 Problémy a zádrhely 3.1 ArrayOutOfBoundsException Jak jsem říkal už při vyplňování pole, tak tato chyba se mi ukazovala kvůli ptaní se na políčka v poli, která ani neexistovala. Dlouho mi trvalo, než jsem přišel na to, co musím udělat, sice název chyby je dost zavádějící ale i tak jsem si nebyl jistý. Tak jsem vytvořil asi šest podmínek, ale všechny pouze pro pozice kde figurovala nula. Takže se mi i přes těchto několik podmínek chyba stále ukazovala. Já jsem však pokračoval v projektu dále, protože se mi i zároveň pole špatně vykreslovalo a zkrátka a jednoduše tato chyba nebyla jediná. Když jsem se však na to opět s odstupem času koukl, uvědomil jsem si, že musím udělat i podmínky pro pozice kde jedna ze souřadnic je devítka, protože jsem si před tím neuvědomil, že i když to neodkazuje na zápornou souřadnici, stále to může být špatně. A tak přibily další podmínky.
Obrázek 5- Jedna z podmínek pro zjištění min kolem
3.2 Označováni a od-označování políček Toto jsem již popsal v ovládání myší, a tak zde nebudu psát o tom, jak tato metoda funguje, nýbrž o tom v čem mi to dělalo problém. Hledal jsem na internetu jak na to, ale i tak jsem nemohl na nic přijít. Jedna z mých myšlenek, na kterou jsem přišel hledáním na internetu, byla taková, že kliknu pravým tlačítkem na políčko a objeví se mi PopUpMenu(menu s možnostmi), ve kterém zvolím označení políčka. To bylo moc složité. Poté jsem zjistil, že existuje metoda BUTTON1 (2,3). A tak jsem označování dal do metody BUTTON2. To mi však stále nefungoval a netušil jsem proč. Důvod byl takový, že jsem si myslel, že BUTTON2 je pravé tlačítko myši. Ono to však bylo prostřední (myslí se tím samozřejmě kolečko). A tak jsem toho využil a do BUTTON2 jsem dal od-označení a do BUTTON3 označení. 3.3 Zobrazování nul Toto je jediný problém, který jsem nedokázal vyřešit. Naštěstí je součástí bonusových funkcí, ale i tak jsem se pokusil toto splnit. Princip této funkce byl takový, že pokud hráč klikne na políčko, které se nedotýká žádné miny, tak se odhalí i všechna okolní. A pokud je na vedlejším místě další políčko s dotykem rovnajícímu se nule, tak se odhalí okolní políčka i u něj a vznikne jakási řetězová reakce. Mě se povedlo pouze zobrazit políčka obklopující odhalené políčko. Využil jsem k tomu již použité podmínky, pro zjištění kolika min se políčko dotýká, pouze jsem je trochu upravil, aby fungovali, jak potřebuji.
Obrázek 6-Rozklikávání v mém programu, rozklikávání v originálních hledání min
Stránka | 6
3.4 Počet min Tento problém jsem se ze začátku snažil vyřešit tím, že nebudu používat pouze náhodu z ano nebo ne, ale tím, že budu každé pole vyplňovat náhodnými čísly 0-9 a pokud by bylo číslo nižší nebo rovno 1 tak by zde byla mina. K tomuto účelu jsem uzpůsobil i všechny podmínky. Poté mi však začalo vadit, že počet min není omezený, a tak jsem začal hledat nový způsob. Udělal jsem cyklus, který se patnáctkrát opakoval a během kterého se vygenerovalo 15 náhodných souřadnic, kde jsou miny umístěny. To však mělo menší zádrhel a to že se mohly objevit dvě miny na jedné souřadnici. A tak k tomu jsem musel vytvořit další cyklus, ve kterém jsem znovu vygeneroval náhodné souřadnice, a poté jsem se ptal, zda tyto souřadnice jsou již minou obsazené, pokud byly, tak se cyklus opakoval.
Obrázek 7-Cyklus k naplňování minami
Stránka | 7
Závěr Upřímně se svým projektem jsem velice spokojený, protože jsem splnil všechny základní cíle, které jsem si dal a to i včetně jednoho z bonusových. Práce jako taková mě bavila a upřímně si myslím, že mi hodně pomohla s programováním jako takovým.
Zdroje http://www.techradar.com/news/gaming/the-most-successful-game-ever-a-history-ofminesweeper-596504
Stránka | 8