UNIVERZA V LJUBLJANI PEDAGOŠKA FAKULTETA
DIPLOMSKO DELO
Dejan Dobnikar
UNIVERZA V LJUBLJANI PEDAGOŠKA FAKULTETA
Študijski program: Matematika in ra č unalništvo
GEOMETRIJSKA PREDSTAVA TRIDIMENZIONALNEGA
PROSTORA S POMO Č JO PRENOSNIH NAPRAV
DIPLOMSKO DELO
Mentor:
prof. dr. Jože Rugelj Kandidat:
Dejan Dobnikar
Ljubljana, november 2012
I
Zahvala
Posebej se zahvaljujem svojemu mentorju, prof. dr. Jožetu Ruglju, za strokovno pomoč in usmerjanje pri izdelavi diplomske naloge.
Zahvaljujem se vsem prijateljem, ki so bili pripravljeni posoditi svoje tablične računalnike, da sem lahko preizkusil delovanje igre na različnih napravah. Prav tako se zahvaljujem vsem, ki so igro preizkušali in podali svoje mnenje o možnih izboljšavah.
Zahvaljujem se prof. Alenki Jurančič, prof. Francki Planinc, prof. Jožici Mlakar Broder in prof. Goranu Vujoviću, da so si vzeli čas, preizkusili igro in podali mnenje.
Zahvala gre tudi dr. Zlatanu Magajni za pomoč pri vsebini igre.
Za konec bi se zahvalil tudi vsem, ki so me spodbujali pri pisanju diplomske naloge.
II
Povzetek
V diplomskem delu se posvetim vprašanju, ali lahko računalniške igre, ki so sicer zelo popularne, povežemo z učenjem. Poizkušam odgovoriti na vprašanja, kot so: Ali pri igrah poteka učenje? Kako pri igrah poteka učenje? Ali bi lahko izdelali igro, ki bi se navezovala na katero izmed poglavij, obravnavanih v šoli, in bila še vedno zanimiva za igranje? Nato je na podlagi tega opisana zasnova, načrt in izdelava igre, ki deluje na napravah z operacijskim sistemom Android.
V uvodu so predstavljeni cilji, v drugem poglavju pa govorim o igrah in povezavi z učenjem. Posebej opišem principe učenja, ki so sicer značilni za učne ure v šoli, najdemo jih pa tudi v računalniških igrah. Tretje poglavje govori o zasnovi igre. Tu predstavim idejo igre ter vse funkcije, ki jih igra vsebuje. V tem poglavju tudi opišem orodja, ki jih potrebujemo za izdelavo Android aplikacij. V četrtem poglavju si pogledamo nekaj posebnosti, ki jih moramo vedeti, preden začnemo z izdelavo aplikacij. Predstavim programski jezik in način izdelave čisto osnovne aplikacije, nato pa se lotim izdelave igre. Za večino delov igre predstavim načrt in izdelavo, na koncu pa te dele še povežemo v celoto, ki je predstavljena v petem poglavju.
Obiskal sem tudi profesorje osnovnih in srednjih šol, da sem pridobil nekaj mnenj o igri. Mnenja so zbrana v šestem poglavju.
Ključne besede: izobraževanje, 3D, prostor, geometrijska, predstava, tridimenzionalni prostor, Android, izdelava, igre, programski jezik, Java, programiranje
ACM klasifikacija:
C. Computer Systems Organization
C.3 SPECIAL-PURPOSE AND APPLICATION-BASED SYSTEMS Real-time and embedded systems: Android naprave
D. Software
D.1 PROGRAMMING TECHNIQUES D.1.5 Object-oriented Programming D.3 PROGRAMMING LANGUAGES
D.3.2 Language Classifications
Object-oriented languages: Java K. Computing Milieux
K.3 COMPUTERS AND EDUCATION Computer-managed instruction (CMI) J. Computer Applications
J.1 ADMINISTRATIVE DATA PROCESSING Education
III
Abstract
The thesis is devoted to the question of whether we can look for parallels between computer games, which are otherwise very popular, and learning. It tries to answer questions such as: Is a learning process present during gaming? How does it happen? Is it possible to make a game that would be connected to certain courses discussed at school and still be interesting to play? Based on the findings, the thesis describes the design, outline and production of a game that works on Android devices.
The introduction presents objectives, while the second chapter speaks about games and their relation to learning. A special emphasis is given to the description of principles of learning that are otherwise characteristic of school lessons, but can also be found in computer games. The third chapter describes the design of the game.
The idea of the game is thoroughly presented, as well as the included functions. This chapter also speaks about the tools needed to create Android applications. The fourth chapter contains some of the features we need to know before making an application. The programming language and the production of a basic application are presented, followed by the process of building the actual game. The thesis then presents the outline and making of various parts of the game, with the organization of these parts into a whole being described in the fifth chapter.
I have also visited elementary and secondary school professors to gather opinions on the game. These are collected in the sixth chapter.
Key words: education, 3D, space, geometric, presentation, three-dimensional space, Android, making of, games, programming language, Java, programming
ACM classification:
C. Computer Systems Organization
C.3 SPECIAL-PURPOSE AND APPLICATION-BASED SYSTEMS Real-time and embedded systems: Android devices
D. Software
D.1 PROGRAMMING TECHNIQUES D.1.5 Object-oriented Programming D.3 PROGRAMMING LANGUAGES
D.3.2 Language Classifications
Object-oriented languages: Java K. Computing Milieux
K.3 COMPUTERS AND EDUCATION Computer-managed instruction (CMI) J. Computer Applications
J.1 ADMINISTRATIVE DATA PROCESSING Education
IV
Kazalo vsebine
1 Uvod ... 1
1 Računalniške igre ... 2
1.1 Igre in učenje ... 3
1.2 Principi učenja v dobrih video igrah ... 4
2 Zasnova igre ... 5
2.1 Liki ... 6
2.2 Načini igre ... 9
2.3 Stopnje in zahtevnost ... 10
2.4 Tretja dimenzija ... 10
2.5 Dodatne zahteve ... 10
2.6 Okolje, kjer bo igra delovala ... 11
2.7 Priprava delovnega okolja – SDK, IDE, ADT ... 13
3 Izdelava igre ... 15
3.1 Programski jezik ... 15
3.2 Predstavitev in posebnosti platforme Android ... 16
3.2.1 Različne verzije sistema Android ... 16
3.2.2 Življenjski cikel programa ... 17
3.2.3 Podpora za več jezikov ... 19
3.2.4 Podpora različnim dimenzijam in gostotam zaslonov ... 20
3.3 Načrt in izdelava ... 21
3.3.1 Glavni meni ... 21
3.3.2 Priprava objektov - Lik ... 22
3.3.3 Priprava objektov – Seznam likov ... 29
3.3.4 Priprava objektov – Sistem preverjanja mrež ... 37
3.3.5 Sestavljanje objektov v celoto ... 40
4 Končni rezultat ... 50
5 Mnenja ... 54
6 Zaključek ... 58
7 Viri ... 58
V
Kazalo slik
Slika 1: igra Tennis for Two iz leta 1958 na osciloskopu [1] ... 2
Slika 2: prikaz števila igralcev pri ponudniku iger Steam [3] ... 3
Slika 3: ideja igre ... 5
Slika 4: liki v igri (posnetek) ... 6
Slika 5: Eclipse IDE ... 13
Slika 6: ADT vtičnik doda tudi pripomoček za uporabniške vmesnike ... 14
Slika 7: virtualna Android naprava ... 15
Slika 8: dedovanje lastnosti razredov ... 16
Slika 9: življenjski cikel programa ... 18
Slika 10: testni program ... 19
Slika 11: struktura projekta ... 20
Slika 12: uporaba dveh različnih uporabniških vmesnikov v istem programu ... 21
Slika 13: diagram glavnega menija ... 21
Slika 14: jezik igre se razlikuje glede na jezik telefona ... 22
Slika 15: definicija razreda Lik in njegovih podrazredov ... 23
Slika 16: vektorski produkt ... 26
Slika 17: lepljenje likov... 32
Slika 18: vrtenje likov ... 36
Slika 19: glavni del igre ... 40
Slika 20: meni igre ... 50
Slika 21: navodilo naloge ... 50
Slika 22: sestavljena mreža piramide ... 50
Slika 23: zapiranje mreže v geometrijsko telo ... 51
Slika 24: sporočilo, da je naloga končana... 51
Slika 25: stopnja z dvema nalogama ... 51
Slika 26: mreža tristrane prizme ... 52
Slika 27: animacija zapiranja tristrane prizme ... 52
Slika 28: možnosti animacije ... 53
Slika 29: neprosojne stranice ... 53
Slika 30: okvir geometrijskega telesa ... 54
Kazalo tabel
Tabela 1: delež operacijskih sistemov pametnih telefonov (podatki v milijonih) [9] .. 12Tabela 2: prodani tablični računalniki (podatki v milijonih) [10] ... 12
Tabela 3: Verzije sistema Android in razširjenost po verzijah [18] ... 16
1
1 Uvod
Prepričan sem, da je že vsakdo opazoval otroka pri igranju računalniških iger.
Marsikdo je tudi opazil, da otroci porabijo za igranje veliko časa, bolj pozorni opazovalci pa so tudi opazili, da so te igre dolge in zahtevne, celo (ali pa še posebej) za odraslega, vendar je vsak otrok pripravljen vložiti svoj čas in se spopasti z izzivi, ki ga čakajo med igranjem. In kar je najpomembneje, pri tem zelo uživa!
Iz tega izhaja tudi moja ideja za diplomsko nalogo, ki sem jo dobil, ko sem opazoval nekega otroka, ki je starše prosil za njihov mobilni telefon, da bi igral igre. Namen diplomske naloge je narediti igro, ki bi posredno ali neposredno pomagala učencem pri katerem izmed obravnavanih poglavij v osnovni ali srednji šoli. Pri tem pa igra ne sme biti samo digitalna oblika učbenika ali delovnega zvezka.
Cilj igre je ozko usmerjen; igra bo poizkušala izboljšati predstavo tridimenzionalnega prostora. To je problem, ki sem ga opazil med prakso v osnovni šoli, predvsem pri individualni pomoči, ter pri inštrukcijah srednješolcev. Ta problem se med drugimi pokaže pri geometriji v prostoru ter kasneje v srednji šoli pri vektorjih v prostoru.
Zaznava tridimenzionalnega prostora pa ni problem samo pri učencih, ampak, kot se je izkazalo pri začetnem testiranju ideje, tudi pri odraslih. Zato ciljna skupina niso samo učenci, ampak vse generacije.
Zasnova igre ter sam potek izdelave sta predstavljena kasneje, najprej pa bom poizkušal ovreči prepričanje (in dokazati nasprotno) mnogih, da so računalniške igre, ki jih igrajo otroci, namenjene samo zabavi, da so izguba časa, da pri igranju iger proces učenja ne poteka ter da igre povečujejo socialno odtujenost.
Ker obstaja tudi zelo široka izbira naprav in operacijskih sistemov, kjer bi igra lahko delovala, bom izbor opisal in tudi utemeljil svojo izbiro.
V diplomski nalogi želim doseči naslednje cilje:
• Izboljšati predstavo tridimenzionalnega prostora s pomočjo računalniške igre
• Izbrati primerne tehnične pripomočke in okolje, kjer bo igra delovala Raziskovalna vprašanja:
• Kako z uporabo IKT izboljšati predstavo tridimenzionalnega prostora
2
1 Ra č unalniške igre
Zgodovina komercialnih video iger sega v sedemdeseta in osemdeseta leta prejšnjega stoletja, prve igre pa so se pojavile v petdesetih letih. Leta 1952 je igro Križci in krožci naredil A. S. Douglas za svoje diplomsko delo, leta 1958 pa je William Higinbotham s pomočjo osciloskopa in analognega računalnika naredil igro za dva igralca Tennis for Two.
Slika 1: igra Tennis for Two iz leta 1958 na osciloskopu [1]
V sedemdesetih in osemdesetih letih so bile popularne video konzole za televizije.
Takrat so se tudi izoblikovali številni žanri iger, kot so akcijske, pustolovske, dirkaške, strateške in simulacijske igre, vendar pa so se igre, kot jih poznamo danes, pojavile v devetdesetih letih, ko so postali procesorji in grafične kartice zmogljivejši in dostopnejši.
S tem se je hitro povečevalo tudi število igralcev, katerih števila ni mogoče natančno določiti, so pa zanimivi nekateri podatki:
• Trenutno ima igra World of Warcraft 9,1 milijona igralcev [2]
• Samo v tem trenutku (17. 8. 2012, 14:10) pri ponudniku iger Steam igra 3.257.271 igralcev sočasno (več različnih iger) [3]
• Povprečna starost igralca iger je 30 let in igra že 12 let [4]
• 47% vseh igralcev je ženskega spola [4]
• 62% vseh igralcev igra igre skupaj z drugimi, osebno ali preko spleta [4]
• 33% jih igra igre na svojih telefonih [4]
• 64% uporabnikov mobilnih telefonov igra igre vsak dan [5]
• 52% igralcev iger na mobilnih telefonih igra več kot eno uro dnevno [5]
• Najpopularnejša igra na mobilnih telefonih, Angry Birds, je od leta 2009 dosegla več kot 700 milijonov prenosov in prinese 6 milijonov dolarjev zaslužka mesečno [5]
3
Slika 2: prikaz števila igralcev pri ponudniku iger Steam [3]
Poudariti je treba, da poleg zgoraj omenjenih iger obstaja na trgu še ogromno drugih iger, ki se sicer ne približajo tem rekordom, a imajo še vedno zelo veliko število igralcev.
Ob zgornjih podatkih lahko zaključimo, da imajo igre nekatere elemente, ki jih naredi zanimive in zaradi katerih igre (vsaj nekatere) zelo počasi izgubljajo število igralcev.
1.1 Igre in u č enje
Čeprav je veliko ljudi mnenja, da so računalniške igre nekoristne ali celo
»zapravljanje časa«, igre pravzaprav vsebujejo veliko lastnosti, ki so pomembne tudi pri učenju. Prva lastnost iger, ki jo opazimo, je moč, s katero igra drži igralca pred računalnikom. Povprečen igralec iger lahko brez premora igra igro tudi po več ur.
Druga pomembna lastnost iger je zahtevnost. Vsakdo, ki je kdaj igral igre, lahko potrdi, da so nekatere igre zelo zahtevne in zaradi tega mora igralec veliko časa tudi trenirati spretnosti, ki jih potrebuje za nadaljevanje. Igra mora imeti pravo ravnovesje zahtevnosti, saj prelahka igra ne nudi izziva, medtem ko pretežka igra prepriča igralca, da napredovanje ni mogoče. Tretja lastnost je nagrajevanje napredovanja.
Ta lastnost je pomembna v povezavi z drugo lastnostjo, saj nagrade pogosto motivirajo igralce, da ne obupajo na stopnji, ki je na videz zahtevna. Poleg tega igre različno nagrajujejo igralce glede na vložen trud in dobljen rezultat, kar je le še dodatna motivacija za igralce. Četrta lastnost iger je takojšnja povratna informacija, zaradi katere igralec takoj ve, kaj je v igri pravilno in kaj ne. To je lastnost, ki je v učilnicah težko dosegljiva zaradi večjega števila učencev. Poleg tega igre prepričajo igralca, da vedno obstaja rešitev ter da je sposoben samostojno rešiti zahtevane naloge.
4
In če nadaljujemo v tej smeri, lahko opišemo nekaj principov učenja v dobrih video igrah.
1.2 Principi u č enja v dobrih video igrah
Najpogostejši principi učenja, ki jih najdemo tudi v video igrah, so: [7]
Princip aktivnega učenja in učenja z refleksijo
Okolje mora biti pripravljeno tako, da spodbuja aktivno učenje ter učenje z refleksijo.
Okolje v igri je lahko poljubno, ustvarimo lahko interaktiven 2D ali 3D svet, lahko je domišljijski, brez omejitev resničnega sveta. Tako ni nobenih omejitev za pripravo okolja, ki bi takšno učenje spodbujal.
Princip vaje
Igralec igre vadi v kontekstu, kjer vaja ni dolgočasna. V dobrih video igrah so tudi pogosto ponavljajoča se dejanja zabavna, in če mora igralec ponavljati določeno nalogo, da v igri napreduje, zakaj ne bi tej nalogi dodali izobraževalnega namena?
Princip dosežkov
Nagrajevanje, prilagojeno učenčevemu znanju in trudu, ki sporoča trenutne dosežke, je v učilnici težko izvedljivo. Video igre pa lahko ponujajo zelo dobro okolje, kjer je to mogoče doseči, saj je lahko igralna izkušnja vsakega igralca unikatna, igralec je v centru svojega napredovanja, igranje oz. reševanje problemov pa ni omejeno s časom. V povezavi s prejšnjim principom pa lahko ponavljajočo se vajo časovno vedno bolj omejujemo.
Režim kompetenc
Učenec dobi veliko priložnosti za delovanje znotraj, a na meji svojih zmožnosti, da dobi občutek, da so stvari zahtevne, a ne neizvedljive.
Princip raziskovanja
Učenje je cikel raziskovanja, refleksije, na podlagi tega formiranje hipoteze, ponovnega raziskovanja in testiranje hipoteze in nato sprejetje ali spreminjanje hipoteze. Ta cikel uporabljamo tudi v dobrih igrah, kjer je treba reševati probleme za napredovanje.
Princip več poti
Vedno obstaja več poti za napredovanje. To omogoča učencem, da se sami odločajo in zanašajo na svoje lastno znanje ter stil učenja in reševanja problemov, lahko pa tudi raziskujejo alternativne stile učenja.
Pri zasnovi igre, ki bi uporabljala ta na igralec ne more rešiti določ
obtiči na isti točki igre. Da pa bi spodbudili igralca, da bi se vra ga poskusil rešiti, bi omogoč
Princip stopnjevanja
Učenje mora biti stopnjevano tako, da za uporabne za kasnejše primere.
znanje ali spretnost, vendar tak na Princip večmodalnosti
Pomen in znanje sta zgrajena skozi razli interakcije, zvok, itd.
Iz zgoraj opisanih principov lahko vidimo, da
tako oddaljene od video igre, ki bi lahko predstavljala odli poglavja, ki jih danes učenci obravnavajo v šoli.
2 Zasnova igre
Namen igre je, kot je bilo omenjeno v uvodu, pomo tridimenzionalnega prostora.
iz osnovnih likov. Uporabnik bo moral izbrati potrebne like, j
mogoče tudi zavrteti, da bodo liki skupaj predstavljali mrežo telesa. Mreže nekaterih likov je mogoče sestaviti na ve
predstava prostora, saj je treba
5
Pri zasnovi igre, ki bi uporabljala ta način, bi lahko dodali možnost, da
igralec ne more rešiti določenega problema, lahko rešuje drug problem in tako ne ki igre. Da pa bi spodbudili igralca, da bi se vračal k temu problemu in kusil rešiti, bi omogočili določene nagrade le v primeru, da reši vse stopnje.
enje mora biti stopnjevano tako, da začetni primeri vodijo do posplošitev, ki so uporabne za kasnejše primere. Podobno velja za igre, kjer je prav tako
znanje ali spretnost, vendar tak način najdemo le v začetnih fazah igre.
Pomen in znanje sta zgrajena skozi različne modalnosti – slike, besedila, simbole,
zgoraj opisanih principov lahko vidimo, da video igre, ki jih ljudje igrajo, sploh niso tako oddaljene od video igre, ki bi lahko predstavljala odlično učno okolje za nekatera
čenci obravnavajo v šoli.
Namen igre je, kot je bilo omenjeno v uvodu, pomoč pri
tridimenzionalnega prostora. Glavna ideja igre temelji na sestavljanju mrež poliedrov Uporabnik bo moral izbrati potrebne like, jih pravilno razporediti in da bodo liki skupaj predstavljali mrežo telesa. Mreže nekaterih e sestaviti na več načinov. Za reševanje teh nalog je potrebna
treba dvodimenzionalni sliki dodati tretjo dimenzijo
Slika 3: ideja igre
in, bi lahko dodali možnost, da v primeru, ko lahko rešuje drug problem in tako ne al k temu problemu in ene nagrade le v primeru, da reši vse stopnje.
do posplošitev, ki so Podobno velja za igre, kjer je prav tako treba osvojiti
etnih fazah igre.
slike, besedila, simbole,
video igre, ki jih ljudje igrajo, sploh niso no okolje za nekatera
č pri prepoznavi Glavna ideja igre temelji na sestavljanju mrež poliedrov ih pravilno razporediti in da bodo liki skupaj predstavljali mrežo telesa. Mreže nekaterih inov. Za reševanje teh nalog je potrebna
tjo dimenzijo.
6
Igra bi morala vsebovati čim več različnih likov, kar bi tudi omogočalo sestavo veliko različnih mrež teles, vendar se bom omejil na šest različnih likov: enakostranična trikotnika dveh velikosti, kvadrata dveh velikosti, enakokraki trikotnik in pravokotnik.
Stranice likov so dolge eno enoto ali enoto in pol.
Slika 4: liki v igri (posnetek)
2.1 Liki
S temi šestimi liki lahko sestavimo naslednje kombinacije teles in mrež:
Prizme
Trije mali kvadrati, dva mala enakostranična trikotnika:
Trije pravokotniki, dva mala enakostranična trikotnika:
7
Dva enakokraka trikotnika, dva pravokotnika, en mali kvadrat:
Dva enakokraka trikotnika, dva velika kvadrata, en pravokotnik:
Trije pravokotniki, dva mala enakostranična trikotnika:
Dva velika enakostranična trikotnika, trije pravokotniki:
8
Dva velika enakostranična trikotnika, trije veliki kvadrati:
Piramide
En mali kvadrat, štirje mali enakostranični trikotniki:
En mali kvadrat, štirje enakokraki trikotniki:
Štirje (mali ali veliki) enakostranični trikotniki:
9
En mali enakostranični trikotnik, trije enakokraki trikotniki:
En pravokotnik, dva velika enakostranična trikotnika, dva enakokraka trikotnika
2.2 Na č ini igre
Predvidenih je več načinov igre. Prvi način je podoben igri s kartami Remi. Igralci dobijo začetne karte, nato pa izmenično kupujejo karte, na katerih so liki. Ko igralec drži karte, iz katerih lahko sestavi mrežo telesa, te karte položi odprte na mizo. Karte, ki so na mizi, lahko igralec vzame tudi nazaj, če lahko iz njih in svojih kart sestavi več teles. Zmaga igralec, ki se prvi znebi kart. Nasprotnik je lahko program sam, ali pa igralec, ki se poveže k igri s pomočjo tehnologije Bluetooth.
Drugi predvideni način igre je časovno neomejeno napredovanje skozi vnaprej pripravljene stopnje. V primeru, da uporabnik ne bo znal rešiti naloge, bo dovoljeno napredovanje po stopnjah, vendar ne več kot dve ali tri stopnje. Tako se bo uporabnik moral vračati in rešiti tudi stopnjo, ki jo je izpustil.
Tretji predvideni način igre je igra na čas. V igri bo navodilo za sestavljanje mrež, uporabnik pa bo moral nalogo izpolniti v danem času.
Četrti način igre, ki sem ga predvidel, pa je večigralski način, ki bo temeljil na drugem in tretjem načinu igre. Igralci bodo tekmovali med seboj, dosežki pa se bodo točkovali glede na uspeh igralcev. Igralec, ki bo prvi zaključil nalogo, bo dobil vse predvidene točke. Od tega trenutka bodo imeli preostali igralci na voljo še nekaj časa, da zaključijo nalogo. Če jim to uspe, bodo nagrajeni z delom točk, ki bo odvisen od časa, ki ga bodo porabili. Uporabniku bo preostal čas viden, saj se bo v ozadju čas odšteval.
10
Režim kompetenc, opisan v prejšnjem poglavju, bi lahko dodali v igro tako, da bi se meril čas reševanja, nato pa bi se na podlagi tega izbirala naslednja stopnja. Primer:
uporabniku, ki bi zelo hitro rešil neko stopnjo, bi igra izbrala težjo naslednjo stopnjo kot uporabniku, ki je porabil veččasa.
Najprej bom implementiral tretji način igre, ki bo imel zaradi demonstracijskih namenov izklopljeno funkcijo odštevanja časa. Nato bom vključil še četrti način igre, saj je ta način najbolj zanimiv zaradi možnosti tekmovanja med različnimi uporabniki.
Funkcija, ki bi upoštevala režim kompetenc, bi zahtevala zelo veliko število nalog, poleg tega bi pa moral še izmeriti povprečje reševanja nalog pri različnih uporabnikih ter določiti standarde za vsako nalogo posebej, poleg tega pa bi moral še upoštevati, da niso vsi uporabniki enako spretni pri uporabi elektronskih naprav.
2.3 Stopnje in zahtevnost
Zahtevnost igre je ključnega pomena. Igra, ki ne predstavlja izziva, ni zanimiva za uporabnike. Prav tako ni zanimiva igra, ki je preveč zahtevna. Zato bodo prve stopnje zelo lahke, nato pa se bo z vsako stopnjo zahtevnost povečala. Nekatere naloge bodo zahtevale samo točno določeno geometrijsko telo ali več teles, druge pa eno ali več točno določenih mrež telesa. Zato bo treba zgraditi sistem, v katerega bo mogoče na enostaven način dodajati in spreminjati stopnje, v ozadju pa bo moral delovati univerzalen sistem preverjanja pravilnosti reševanja stopenj.
2.4 Tretja dimenzija
Da bo igra omogočala predstavo tridimenzionalnega prostora, bo potrebna animacija, ki bo jasno prikazala, kako se liki, ki so narisani v dveh dimenzijah, zavrtijo, da nastane polieder. Za to bo potrebna transformacija iz dvodimenzionalnega prostora, v katerem se bodo sestavljale mreže teles, v tridimenzionalni prostor, v katerem bodo potekale animacije zapiranje teles.
2.5 Dodatne zahteve
V dvodimenzionalnem pogledu bo treba like zložiti skupaj, da bodo predstavljali mrežo telesa. Liki bodo morali biti natančno zloženi eden ob drugem. Da bo to mogoče, bodo morali biti liki »lepljivi«: Lik, ki se bo približal drugemu liku dovolj blizu, se bo moral sam poravnati. Da bo možno zlepiti med seboj kvadrat in trikotnik, bo potrebna tudi možnost rotacije likov.
Potreben bo tudi sistem preverjanja sestavljenih likov. Vsakič, ko bomo zlepili med seboj like, bo treba preveriti, ali sestavljeni liki že predstavljajo kako mrežo telesa.
11
Potrebno bo razpoznavanje vsake mreže posebej, da bomo iz nje sestavili podatke o animaciji, ki se bo izvajala v tridimenzionalnem načinu. Animacija bi se morala odvijati v obe smeri – lik bi se moral tako zapirati kot odpirati, z možnostjo začasne ustavitve animacije. Potrebne bodo tudi dodatne črte ali predmeti, ki bi pomagali pri zaznavi prostora.
V tridimenzionalnem načinu je predvidena tudi možnost povečave in obračanja pogleda, da bo postavitev likov oz. stranic telesa jasno prikazana. Interakcija, ki bo potekala s prsti na zaslonih, občutljivih na dotik, bo morala biti zelo intuitivna, da bo uporaba čim bolj preprosta. Možna bo uporaba več dotikov naenkrat, npr. pri povečavi.
Sama igra zasnova igre bi morala biti modularna, da bi bilo dodajanje novih likov in nalog možno s posodabljanjem igre in ne s ponovnim prenosom, saj je v prihodnosti predvideno dodajanje novih likov in novih nalog. Igra bo morala biti tudi večjezično zasnovana, da bo mogoča čim širša uporaba igre.
2.6 Okolje, kjer bo igra delovala
Možnost izbire okolja, v katerem bi igra delovala, je velika. Na področju namiznih in prenosnih računalnikov se najpogosteje uporabljajo operacijski sistemi Windows (Microsoft), Linux (odprtokodni operacijski sistem) in Mac OS (Apple). Na področju tabličnih računalnikov in pametnih telefonov pa so najbolj razširjeni operacijski sistemi Android (Google), iOS (Apple), Windows Phone (Microsoft), manj pa Symbian (Nokia), Bada (Samsung) in Blackberry OS (RIM).
Odločil sem se, da bo igra delovala na pametnih telefonih in tabličnih računalnikih.
Razlogov za to je več: pametni telefoni in tablični računalniki si operacijski sistem delijo (Android, iOS), zato aplikacije delujejo na obeh tipih naprav. Te naprave omogočajo boljšo interakcijo, saj računalniško miško nadomestijo dotiki zaslona, ki lahko zaznavajo do deset dotikov naenkrat. Ker so vgrajeni senzorji, je možna uporaba naprave glede na naklon in nagib, tablični računalniki imajo v primerjavi s prenosnimi računalniki nekajkrat boljšo avtonomijo, so veliko lažji in manjši ter zato tudi lažje prenosljivi.
Poleg tega mora biti operacijski sistem tudi čim bolj razširjen, da ima čim več uporabnikov teh naprav možnost preizkusa igre.
12
Operacijski sistem
Q2 2012 Pošiljk na
trg
Q2 2012 Tržni delež
Q2 2011 Pošiljk na
trg
Q2 2011 Tržni delež
Sprememba
Android 104.8 68.1% 50.8 46.9% 106.5%
iOS 26.0 16.9% 20.4 18.8% 27.5%
BlackBerry OS 7.4 4.8% 12.5 11.5% -40.9%
Symbian 6.8 4.4% 18.3 16.9% -62.9%
Windows Phone 7 5.4 3.5% 2.5 2.3% 115.3%
Linux 3.5 2.3% 3.3 3.0% 6.3%
Ostali 0.1 0.1% 0.6 0.5% -80.0%
Skupaj 154.0 100.0% 108.3 100.0% 42.2%
Tabela 1: delež operacijskih sistemov pametnih telefonov (podatki v milijonih) [9]
Tabela 2: prodani tablični računalniki (podatki v milijonih) [10]
Iz teh tabel lahko razberemo, da je operacijski sistem Android v prednosti pri mobilnih telefonih, medtem ko je pri 68,7 milijona prodanih tabličnih računalnikov v letu 2011v prednosti podjetje Apple s sistemom iOS.
Če primerjamo še cene tabličnih računalnikov, so v prednosti naprave, na katerih deluje operacijski sistem Android. Ker so te naprave dostopnejše, so v splošnem tudi bolj primerne za širšo uporabo, npr. v šolah in podobnih ustanovah, kjer je treba zagotoviti vsakemu uporabniku svoj tablični računalnik. Pri tem pa je treba še preveriti, ali za tablične računalnike, ki jih nameravamo kupiti, obstaja programska oprema, ki je ključna za naše delo.
Zaradi razširjenosti in dostopnosti naprav sem se odločil, da bo igra delovala na pametnih telefonih in tabličnih računalnikih s sistemom Android.
13
2.7 Priprava delovnega okolja – SDK, IDE, ADT
Preden začnemo z delom, moramo pripraviti delovno okolje in namestiti vse potrebno za razvijanje programske opreme za platformo Android.
SDK
SDK je kratica za software development kit, ali po slovensko komplet programskih orodij za razvijanje programske opreme. Namestiti moramo dva, in sicer najprej Java JDK [12], nato pa še Android SDK [13]. Oba paketa dobimo brezplačno na spletnih straneh [12][13].
IDE
IDE je kratica za integrated development environment, ali po slovensko razvojno okolje. Je programska oprema, ki jo uporabljamo za razvijanje programov. Najbolj razširjeno razvojno okolje za programski jezik Java je Eclipse, ki ga moramo prenesti iz uradne spletne strani [14] in namestiti. Eclipse je odprtokodno razvojno okolje in zaradi tega brezplačno.
Za programski jezik Java so na voljo tudi druga razvojna okolja, vendar je za Eclipse na voljo vtičnik (plugin), ki nam olajša razvoj Android aplikacij.
Slika 5: Eclipse IDE
14
ADT
Android Development Tools [15] je vtičnik za Eclipse IDE. ADT razširi zmožnosti razvojnega okolja Eclipse in nam omogoča pripravo Android projektov, pomaga zgraditi uporabniški vmesnik aplikacije in omogoči razhroščevanje aplikacij (angleško debugging) [17]. Poleg tega vsebuje še virtualno Android napravo oz. Android emulator, s katero lahko testiramo svojo aplikacijo, če nimamo na voljo telefona ali tabličnega računalnika. Uporabimo ga lahko za simulacijo klicanja in pošiljanja SMS sporočil in za simulacijo prenosa podatkov. Virtualna naprava ima nekaj pomanjkljivosti, kot so počasno delovanje, nezmožnost zaznavanja več dotikov hkrati in pomanjkanje podpore tehnologiji Bluetooth, in zaradi tega v nekaterih primerih ni primerna za resno testiranje, vendar je v veliko pomoč možnost nastavljanja različnih dimenzij zaslonov. Tako lahko preverimo, kako bi bil program videti na drugih napravah. To je zelo pomembno pri razvijanju aplikacij, pri katerih moramo sami upoštevati število točk na zaslonu.
Celoten seznam funkcij, ki niso podprte v virtualni napravi, lahko najdete na spletni strani developer.android.com [16].
Slika 6: ADT vtičnik doda tudi pripomoček za uporabniške vmesnike
15
Slika 7: virtualna Android naprava
Ko imamo vse to nameščeno, smo pripravljeni za delo.
3 Izdelava igre
3.1 Programski jezik
Za programiranje Android aplikacij uporabljamo programski jezik Java, možna pa je tudi uporaba programskih jezikov C ali C++, ki se uporabljata (za celotno aplikacijo ali pa samo del nje), kadar želimo doseči veliko hitrost izvajanja programov. A glede na dokumentacijo [11] njuna uporaba ne prinese vedno opazne pohitritve, poveča pa kompleksnost aplikacije. Priporočljiva je uporaba za operacije, ki zelo obremenijo CPE (centralno procesno enoto), npr. za procesiranje signalov, simulacijo vpliva fizike in za stiskanje slik.
Java je programski jezik, ki podpira objektno orientiran pristop k programiranju. Tak pristop nam omogoča modularno strukturo programiranja in preprosto ponovno uporabo obstoječe kode. Pri objektno orientiranem programiranju uporabljamo objekte kot instance razredov, kjer so definirane spremenljivke in metode. Razredi lahko dedujejo lastnosti drugih razredov. Spremenljivke in metode, definirane v razredih, so lahko dveh vrst: statične in dinamične. Dinamične spremenljivke
16
pripadajo vsaki instanci razreda posebej, medtem ko so statične spremenljivke skupne vsem instancam razredov. Podobno je s statičnimi in dinamičnimi metodami.
Slika 8: dedovanje lastnosti razredov
Na zgornji sliki imamo štiri razrede. Razred Vozilo vsebuje podatka o barvi vozila in o imenu lastnika, vsebuje pa tudi metodo za zagon motorja. Razredi Avto, Tovornjak in Motorno kolo razširjajo razred Vozilo. To pomeni, da razred Avto deduje lastnosti razreda Vozilo in poleg podatkov o številu vrat in velikosti prtljažnika vsebuje še podatka o barvi in lastniku ter metodo za zagon motorja.
3.2 Predstavitev in posebnosti platforme Android
3.2.1 Razli č ne verzije sistema Android
Pri izdelavi programske opreme za platformo Android je treba poznati nekaj posebnosti, ki se nanašajo na mobilne naprave. Najprej moramo paziti na veliko število verzij sistema Android, saj se ta zelo hitro razvija.
Trenutna verzija je Android 4.1, spodnja tabela pa prikazuje razširjenost po verzijah.
Verzija Kodno ime API Razširjenost
(4. September 2012) Razširjenost (1. maj 2012)
1.5 Cupcake 3 0,2% 0,3%
1.6 Donut 4 0,4% 0,7%
2.1 Eclair 7 3,7% 5,5%
2.2 Froyo 8 14% 20,9%
2.3 – 2.3.2 Gingerbread 9 0,3% 0,5%
2.3.3 – 2.3.7 10 57,2% 63,9%
3.1 Honeycomb 12 0,5% 1,1%
3.2 13 1,6% 2,2%
4.0 – 4.0.2 Ice Cream Sandwich 14 0,1% 0,5%
4.0.3 – 4.0.4 15 20,8% 4,4%
4.1 Jelly Bean 16 1,2% Izšel julija 2012
Tabela 3: Verzije sistema Android in razširjenost po verzijah [18]
17
Igra mora biti kompatibilna s čim večjim številom naprav, zato bomo pazili, da bo delovala vsaj na verziji 2.2 in kasnejših. S tem bo igra delovala na 95% vseh naprav.
Igra bo lahko delovala tudi na verziji 2.1, vendar zaradi omejene podpore sledenju več dotikom na nekaterih napravah med razvijanjem igre ne bom osredotočen na to verzijo.
3.2.2 Življenjski cikel programa
Naslednja stvar, na katero moramo biti pozorni, je življenjski cikel programa.
Osnovni gradnik, tisto, kar vidimo na zaslonu, se v platformi Android imenuje Activity [19]. Vsak program, ki vsebuje tudi uporabniški vmesnik, vsebuje tudi vsaj en Activity, ki skrbi za življenjski cikel programa ter za kreiranje okna in postavitev uporabniškega vmesnika, ki ga določimo, v to okno.
Pot do prvega programa je preprosta, v razvojnem okolju Eclipse ustvarimo nov Android projekt, v katerega dodamo svoj razred, ki razširja razred Activity (rečemo lahko tudi, da je ta razred podrazred razreda Activity).
public class myActivity extends Activity { @Override
public void onCreate(Bundle savedInstanceState) {// OBVEZNO
super.onCreate(savedInstanceState); // KLIC METODE onCreate V RAZREDU ACTIVITY
setContentView(R.layout.activity_my); // DOLOČIMO UPORABNIŠKI VMESNIK }
// PRIMERNO ZA SHRANJEVANJE PODATKOV @Override
public void onPause() { super.onPause();
}
// PRIMERNO ZA BRANJE SHRANJENIH PODATKOV @Override
public void onResume() { super.onResume();
}
@Override
public void onRestart() { super.onRestart();
}
@Override
public void onStart() { super.onStart();
}
@Override
public void onStop() { super.onStop();
}
@Override
public void onDestroy() { super.onDestroy();
} }
V naš razred moramo vključ
ob zagonu in kreiranju programa. V
vmesnik, ki si ga bomo ogledali kasneje. Priporo
onPause(); tu je potrebno shranjevati podatke, ki bi jih ob programa spet potrebovali.
programa, so razložene v naslednjem diagramu.
spremembi življenjskega cikla, moramo poklicati tudi istoimensko metodo v glavn razredu.
18
V naš razred moramo vključiti vsaj metodo onCreate(), kjer določ ob zagonu in kreiranju programa. V tem primeru samo določ vmesnik, ki si ga bomo ogledali kasneje. Priporočljivo je vključ
tu je potrebno shranjevati podatke, ki bi jih ob naslednjem zagonu programa spet potrebovali. Vse te metode, ki se izvajajo skozi življenjski cikel programa, so razložene v naslednjem diagramu. V vsaki metodi, ki
spremembi življenjskega cikla, moramo poklicati tudi istoimensko metodo v glavn
Slika 9: življenjski cikel programa
kjer določimo kaj se zgodi samo določimo uporabniški ljivo je vključiti tudi metodo naslednjem zagonu skozi življenjski cikel V vsaki metodi, ki se izvede ob spremembi življenjskega cikla, moramo poklicati tudi istoimensko metodo v glavnem
Uporabniški vmesniki so obi primeru activity_my.xml:
<RelativeLayout xmlns:android
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
<TextView
android:layout_width android:layout_height
android:layout_centerHorizontal android:layout_centerVertical
android:padding="@dimen/padding_medium"
android:text="@string/hello_world"
tools:context=".myActivity"
</RelativeLayout>
Če preizkusimo zgornji program, je na telefonu
3.2.3 Podpora za ve č
Podpora za več jezikov je pri sistemu Android zelo preprosta.
shranjevati v datoteko strings.xml
naslednji strani), gradniki uporabniškega vm besedila v datoteki strings.xml
besedila v privzetem jeziku, npr. v angleš slovenski jezik, ustvarimo mapo z imenom
strings.xml, v kateri so slovenski prevodi. Podobno lahko ustvarimo mapo values in ji dodamo
3166-1. Program bo sam prepoznal jezik telefona in uporabljal ustrezne prevode, pa prevodi ne obstajajo, uporablja privzeta besedila, shranjena v mapi
Primer: Referenca do besedila v datoteki uporabniškega vmesnika):
19
običajno definirani v datotekah s končnico activity_my.xml:
xmlns:android="http://schemas.android.com/apk/res/android"
"http://schemas.android.com/tools"
"match_parent"
"match_parent" >
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerHorizontal="true"
android:layout_centerVertical="true"
"@dimen/padding_medium"
"@string/hello_world"
".myActivity" />
izkusimo zgornji program, je na telefonu videti tako:
Slika 10: testni program
Podpora za ve č jezikov
jezikov je pri sistemu Android zelo preprosta. Vsa besedila moramo strings.xml v mapi values (glej sliko struktu
), gradniki uporabniškega vmesnika pa morajo vsebovati referenco do strings.xml Priporočljivo je, da so v mapi values
besedila v privzetem jeziku, npr. v angleščini. Če pa želimo dodati prevode za slovenski jezik, ustvarimo mapo z imenom values-sl, v katero shranimo datoteko , v kateri so slovenski prevodi. Podobno lahko za katerikoli jezik
in ji dodamo dvočrkovno oznako države po standardu . Program bo sam prepoznal jezik telefona in uporabljal ustrezne prevode, pa prevodi ne obstajajo, uporablja privzeta besedila, shranjena v mapi
Primer: Referenca do besedila v datoteki strings.xml : android:text="@string/hello_world"
čnico .xml, v našem
"http://schemas.android.com/apk/res/android"
Vsa besedila moramo (glej sliko strukture projekta na esnika pa morajo vsebovati referenco do values shranjena e pa želimo dodati prevode za ero shranimo datoteko za katerikoli jezik države po standardu ISO . Program bo sam prepoznal jezik telefona in uporabljal ustrezne prevode, če pa prevodi ne obstajajo, uporablja privzeta besedila, shranjena v mapi values.
(Glej definicijo
20 Vsebina datoteke strings.xml v mapi values:
<resources>
<string name="app_name">Hello, Android</string>
<string name="hello_world">Hello world!</string>
<string name="menu_settings">Settings</string>
<string name="title_activity_my">myActivity</string>
</resources>
Vsebina datoteke strings.xml v mapi values-sl:
<resources>
<string name="app_name">Pozdrav, Android</string>
<string name="hello_world">Pozdrav!</string>
<string name="menu_settings">Nastavitve</string>
<string name="title_activity_my">myActivity</string>
</resources>
3.2.4 Podpora razli č nim dimenzijam in gostotam zaslonov
Zadnja stvar, na katero bomo morali biti pozorni, so različne dimenzije in gostote zaslonov. Da se pravilno lotimo tega problema, je potrebno upoštevati dokumentacijo na strani developer.android.com [20][21][22]. Uporabljati moramo enote, ki niso odvisne od velikosti in števila pik na zaslonu. Prva enota se imenuje density-independent pixel (dp), ali po slovensko točka, neodvisna od gostote zaslona. Na zaslonu z gostoto 160 dpi (Dots Per Inch ali 160 točk na 2,54 cm) se ena točka ujema z enoto 1dp, na zaslonu z gostoto 320 dpi pa 1dp predstavlja dve točki na zaslonu.
Druga enota se imenuje scale-independent pixel (sp).
To enoto je priporočljivo uporabljati pri tekstu, saj se spreminja glede na velikost pisave, ki jo uporabnik izbere na telefonu ali tabličnem računalniku.
Poleg uporabe pravilnih enot pa si lahko pomagamo tudi z uporabo različnih uporabniških vmesnikov. V projektu moramo shraniti različne uporabniške vmesnike v različne mape, kot je prikazano na sliki levo. Program ob zagonu samodejno določi uporabniški vmesnik glede na velikost in orientacijo naše naprave. V mapi layout shranimo privzeti uporabniški vmesnik. V mapo layout-land pa shranimo uporabniški vmesnik, ki je
Slika 11: struktura projekta
21
namenjen za uporabo, če je naprava v ležečem položaju. Če želimo, da naš program uporabi drugačen uporabniški vmesnik za naprave, ki imajo večje zaslone (tablični računalniki), shranimo uporabniški vmesnik v mapo layout-large.
Slika 12: uporaba dveh različnih uporabniških vmesnikov v istem programu
Na zgornji sliki vidimo, kako lahko izkoristimo zaslon glede na položaj naprave.
3.3 Na č rt in izdelava
Ker bo izdelava igre zahtevna, bom potreboval načrt. Igro bom razbil na posamezne dele in za vsak del najprej predstavil načrt, nato pa izvedbo.
3.3.1 Glavni meni
Najprej potrebujem grob načrt glavnega menija.
Slika 13: diagram glavnega menija
22
Po premisleku sem se odločil za postavitev glavnega menija tako, kot je prikazano v diagramu zgoraj. V meniju Nastavitve bodo različne možnosti, npr. nastavitev jezika, nastavitev velikosti pisav in ostale nastavitve, po katerih se bo pojavila potreba kasneje. Način za več igralcev bom dodal v meni takoj, vendar bo deloval le v primeru, če bo ostalo dovolj časa za izvedbo.
Izvedbe menija ne bom predstavil, ker ni pomemben za igranje igre.
Slika 14: jezik igre se razlikuje glede na jezik telefona
3.3.2 Priprava objektov - Lik
Zaslon, na katerem igramo igro, je glavni del igre in zato tudi najbolj kompleksen.
Sestavljen mora biti iz dela, kjer se bodo prikazovali liki, ki so na voljo za sestavljanje, predvideni pa so tudi trije gumbi. Preostanek prostora pa bo namenjen za sestavljanje likov v mreže teles.
Prvi gumb bo odprl nov meni, kjer bodo gumbi za izhod v glavni meni, ponovno reševanje stopnje ter za nadaljevanje igre. V tem meniju si bo mogoče ponovno ogledati pomoč, kjer bo razložen potek igre. Drugi gumb bo namenjen ponovnemu prikazu navodila, tretji gumb bo pa imel funkcijo preverjanja pravilnosti sestavljene mreže. Če bo mreža pravilna, se bo zagnala animacija zapiranja likov v telo, sicer pa se bo prikazalo obvestilo o napačnem odgovoru.
Pri likih želimo čim bolje izkoristiti prednosti objektno orientiranega programskega jezika. Smiselno je definirati abstraktni razred Lik, v katerem bodo vse spremenljivke in metode, ki bodo vsem likom skupne. Nato pa bomo definirali
23
konkretne like, ki bodo razširjali razred Lik in jih bomo tudi uporabili kot objekte.
Taka zasnova nam bo omogočala, da ob morebitnih spremembah spreminjamo samo razred Lik, ne pa tudi vse podrazrede.
Za spremenljivke in metode, ki jih bomo uporabili v razredu Lik in njegovih podrazredih, si poglejmo naslednjo sliko.
Slika 15: definicija razreda Lik in njegovih podrazredov
Poleg zgornjih spremenljivk in metod v razredu Lik je pomembna še ena konstanta:
dolžina ene enote v točkah na zaslonu naprave. Ker se dimenzije zaslonov od naprave do naprave spreminjajo, bomo določili, da bo dolžina stranice enaka 80 dp, kar pomeni 80 točk na zaslonu s 160 točkami na 2,54 cm. Ob zagonu programa
24
bomo preračunali 80 dp v število točk na zaslonu in jih zapisali v razred Lik. To konstanto bomo potem uporabljali za računanje oglišč likov. Tako bomo dosegli, da bodo liki na vseh napravah enako veliki.
Sedaj si poglejmo odseke kode iz razreda Lik.
Rotacija točk
Pri rotaciji točke okoli točke si pomagamo z vektorji in matrikami:
′′ = cos ∝ − sin ∝ sin ∝ cos ∝
Pri tem moramo paziti, da zgornji način zavrti točko (, ) za kot ∝ okoli koordinatnega izhodišča. Če želimo točko A zavrteti okoli točke B, ki ni v koordinatnem izhodišču, moramo od točke A odšteti točko B, nato točko A zavrteti okoli koordinatnega izhodišča, nato pa točki A prišteti točko B.
/**
* Zavrti točko okoli točke za določen kot - v stopinjah
*
* @param pointToRotate
* @param pointToRotateAround
* @param angle
* @return
*/
public Coordinates rotatePointAroundPoint(Coordinates pointToRotate, Coordinates pointToRotateAround, float angle)
{
if (Float.isNaN(angle))
{ // če angle neveljavno število, vrnemo točko brez rotacije return pointToRotate;
}
// stopinje pretvorimo v radiane
float angleInRadians = (float) Math.toRadians(angle);
// začasni objekti
Coordinates tmp = new Coordinates();
Coordinates returnPoint = new Coordinates();
// prestavimo v središče koordinatnega sistema tmp.x = pointToRotate.x - pointToRotateAround.x;
tmp.y = pointToRotate.y - pointToRotateAround.y;
returnPoint.x = (float) (tmp.x * Math.cos(angleInRadians) - tmp.y * Math.sin(angleInRadians) );
returnPoint.y = (float) (tmp.x * Math.sin(angleInRadians) + tmp.y * Math.cos(angleInRadians) );
// prestavimo iz središča nazaj
returnPoint.x += pointToRotateAround.x;
returnPoint.y += pointToRotateAround.y;
return returnPoint;
}
V zgornji metodi smo rešili problem vrtenja to uporabimo za vrtenje lika okoli centra ali poljubne to Vrtenje lika okoli centra:
/**
* Rotacija okoli centra, v stopinjah.
* @param angle
*/
public void rotateAroundCenter(
{
// vertexArray je tabela, v kateri so zapisana oglišča for (int i = 0; i <
{
vertexArray[i] = centerCoordinates, angle);
} }
Vrtenje lika okoli poljubne toč
public void rotateAroundPoint(
{
for (int i = 0; i <
{
vertexArray[i] = rotatePointAroundPoint(
}
// poleg oglišč zavrtimo še središče lika centerCoordinates = rotatePointAroundPoint(
}
Preverjanje, ali je točka znotraj lika
To preverjanje uporabimo za
ali je točka znotraj lika, je za pravokotnike
Tu je treba samo preveriti, da je 2, ter da je koordinata toč
Če pa kvadrat zavrtimo, to preverjanje ni ve preverjanja ni primeren za trikotnike ali pet
25
V zgornji metodi smo rešili problem vrtenja točke okoli točke. To lahko sedaj uporabimo za vrtenje lika okoli centra ali poljubne točke.
* Rotacija okoli centra, v stopinjah.
rotateAroundCenter(float angle)
vertexArray je tabela, v kateri so zapisana oglišča i = 0; i < vertexArray.length; i++)
[i] = rotatePointAroundPoint(vertexArray[i], , angle);
koli poljubne točke:
rotateAroundPoint(float angle, Coordinates point) i = 0; i < vertexArray.length; i++)
[i] = rotatePointAroundPoint(vertexArray[i], point, angle);
poleg oglišč zavrtimo še središče lika
= rotatePointAroundPoint(centerCoordinates
Preverjanje, ali je točka znotraj lika
za določanje, ali se s prstom dotaknemo ka znotraj lika, je za pravokotnike preprosta:
da je koordinata točke med koordinatama to koordinata točke med koordinatama točk 0 in 3
e pa kvadrat zavrtimo, to preverjanje ni več tako preprosto. Prav
trikotnike ali petkotnike, če jih bomo kasneje dodali v igro.
čke. To lahko sedaj
[i],
[i], point, angle);
centerCoordinates, point, angle);
dotaknemo lika. Preverjanje,
koordinatama točk 3 in 3.
tako preprosto. Prav tako ta način e jih bomo kasneje dodali v igro.
Za rešitev tega problema
Uporabili bomo lastnost vektorskega produkta:
Recimo, da imamo v ravnini
pa nas zanima, ali leži na isti strani dal vektor in .
vektorja in enako sm imata vektorja in
To lastnost nato uporabimo v metodi razreda Lik:
// metoda vrne TRUE, če sta točki P1 in P2 na isti strani daljice AB protected boolean sameSide(Coordinates P1, Coordinates P2, Coordinates A, Coordinates B) {
MyVector v0 = new MyVector(B.
MyVector v1 = new MyVector(P1.
MyVector v2 = new MyVector(P2.
MyVector cp1 = crossProduct(v0, v1);
MyVector cp2 = crossProduct(v0, v2);
if (dotProduct(cp1, cp2) >= 0) return true;
}
else {
return false;
} }
26
lahko uporabimo znanje, pridobljeno pri urah algebre.
lastnost vektorskega produkta:
v ravnini točke , !, in . Točke , ! in poznamo, za to pa nas zanima, ali leži na isti strani daljice !"""" kot točka . Imejmo vektor
Če sta točki in na isti strani daljice enako smer, če pa sta točki in na različ
različno smer.
Slika 16: vektorski produkt
To lastnost nato uporabimo v metodi razreda Lik:
metoda vrne TRUE, če sta točki P1 in P2 na isti strani daljice AB sameSide(Coordinates P1, Coordinates P2, Coordinates A,
MyVector(B.x - A.x, B.y - A.y, 0);
MyVector(P1.x - A.x, P1.y - A.y, 0);
MyVector(P2.x - A.x, P2.y - A.y, 0);
MyVector cp1 = crossProduct(v0, v1);
MyVector cp2 = crossProduct(v0, v2);
(dotProduct(cp1, cp2) >= 0) {
lahko uporabimo znanje, pridobljeno pri urah algebre.
poznamo, za točko . Imejmo vektor !, na isti strani daljice !"""", potem imata na različnih straneh, pa
metoda vrne TRUE, če sta točki P1 in P2 na isti strani daljice AB sameSide(Coordinates P1, Coordinates P2, Coordinates A,
27
S pomočjo te metode v razredu Lik lahko v podrazredih ugotovimo, ali je dana točka znotraj lika.
Primer iz razreda EnakostranicniTrikotnik.java:
@Override
public boolean isPointInside(Coordinates c) {
if ( sameSide(c, vertexArray[0], vertexArray[1], vertexArray[2]) &&
sameSide(c, vertexArray[1], vertexArray[0], vertexArray[2]) &&
sameSide(c, vertexArray[2], vertexArray[0], vertexArray[1]) ) {
return true;
} else {
return false;
} }
Zgornja metoda preveri, ali je za vsako stranico točka na isti strani kot oglišče trikotnika, ki ne leži na stranici. Če so vsi pogoji izpolnjeni, točka leži znotraj lika.
Podrazredi razreda Lik
Podrazredi razreda lik morajo vsebovati abstraktne metode razreda Lik, poleg tega pa mora biti v vsakem podrazredu podan način, kako se izračunajo oglišča lika. To naredimo v konstruktorju objekta:
public EnakostranicniTrikotnik(int centerX, int centerY) {
super(); // klic konstruktorja razreda Lik
…
centerCoordinates.x = centerX; // določitev centra lika centerCoordinates.y = centerY;
vertexArray = new Coordinates[3]; // ustvarimo novo tabelo dolžine 3 vertexArray[0] = new Coordinates();
vertexArray[0].x = centerX; // izračun prve točke
vertexArray[0].y = (float) (centerY - ( (Lik.oneUnit * 1.73205) / 3));
vertexArray[1] = new Coordinates(); // drugo točko določimo z rotacijo prve vertexArray[1] = rotatePointAroundPoint(vertexArray[0], centerCoordinates, 120);
vertexArray[2] = new Coordinates();// tretjo točko določimo z rotacijo prve vertexArray[2] = rotatePointAroundPoint(vertexArray[0], centerCoordinates, 240);
… }
Vedno, ko bomo ustvarili nov objekt razreda EnakostranicniTrikotnik, se bo izvedla zgornja koda.
Ostala nam je še metoda za izris samega lika. Ta mora za risanje prejeti parameter tipa Canvas [23], ki določa risalno površino, na katero rišemo.
28
public void draw(Canvas canvas) {
path.rewind(); // točke lika dodamo v objekt Path path.moveTo(vertexArray[0].x, vertexArray[0].y);
path.lineTo(vertexArray[1].x, vertexArray[1].y);
path.lineTo(vertexArray[2].x, vertexArray[2].y);
path.close();
if (scaleWhenDrawing) // pomanjšan lik se uporablja ko lik ni na delovni površini {
scaleMatrix.setScale(scale, scale, centerCoordinates.x, centerCoordinates.y);
scaledPath.reset();
scaledPath.set(path);
scaledPath.transform(scaleMatrix); // pomanjšamo
if (!Lik.drawFrameOnly) // če ne rišemo samo okvirja, narišemo najprej polnilo {
canvas.drawPath(scaledPath, fillPaint);
}
canvas.drawPath(scaledPath, paint); // nato pa še rob }
else
{ // rišemo lik v polni velikosti if (!Lik.drawFrameOnly)
{
canvas.drawPath(path, fillPaint);
}
canvas.drawPath(path, paint);
}
// pomoč za razhroščevanje; za delo z rotacijami in za obnašanje likov med spajanjem if (debugModeEnabled)
{
canvas.drawLine(centerCoordinates.x, centerCoordinates.y,
vertexArray[rotationReferenceVertex].x, vertexArray[rotationReferenceVertex].y, paint);
canvas.drawLine(centerCoordinates.x, centerCoordinates.y, LikHolder.refX, LikHolder.refY, paint);
if (showSidesLength) {
canvas.drawText("1", centerCoordinates.x -
(Lik.oneUnit/10),centerCoordinates.y + (Lik.oneUnit/10), textPaint);
}
if (isPressed) {
canvas.drawText("P!", centerCoordinates.x,centerCoordinates.y, textPaint);
}
if (gluedToID != -1) {
canvas.drawText("G:" + gluedToID, centerCoordinates.x, centerCoordinates.y, textPaint);
} }
}
Na podoben način ustvarimo še razrede za kvadrata dveh velikosti, pravokotnik in enakokraki trikotnik.
H kreiranju likov in klicanju metode draw(Canvas canvas) se bomo vrnili še kasneje, ko bomo like vključili v igro.
29
3.3.3 Priprava objektov – Seznam likov
Naslednja stvar, ki jo potrebujemo, je razred, v katerem bo seznam likov, ki bodo na glavni površini. V ta seznam bomo dodajali like, ki bodo na začetku igre ob robu zaslona. S pomočjo tega razreda bomo preverjali, ali sta dva lika tako blizu, da jih lahko zlepimo. Razred bo skrbel tudi za to, da če premikamo en lik po zaslonu, se premikajo tudi vsi liki, ki so k temu liku prilepljeni. Poleg tega bo izvajal rotacije likov, ki jih držimo s prsti.
Javne metode, ki jih bomo potrebovali v tem razredu, so naslednje:
• Metoda za dodajanje lika v seznam
• Metoda, ki preveri, kateri liki so prilepljeni k danemu liku in jih označi za premikanje
• Metoda za preverjanje sestavljenih mrež
• Metoda, ki odstrani referenčne točke za vrtenje
• Metoda, ki odstrani oznake likov za premikanje
• Metoda, ki doda referenčno točko za rotacije likov
• Metoda, ki izbriše določen lik iz seznama in ga vrne
• Metoda, ki izbriše ves seznam
• Metoda, ki nariše like
• Metoda, ki vrne lik z danim ID
• Metoda, ki vrne število elementov v seznamu
• Metoda, ki preveri, ali smo pritisnili lik in vrne ID
• Metoda, ki premakne lik skupaj z liki, ki so zlepljeni
• Metoda, ki odstrani likom oznako, da ga držimo
• Metoda, ki posodobi referenčno točko za rotacijo in like zavrti
Poleg javnih metod bomo potrebovali tudi privatne metode. Do teh ni mogoče dostopati izven razreda, saj so namenjene za pomoč javnim metodam.
Privatne metode:
• Metoda, ki liku izračuna kot med vektorjem, ki poteka od središča lika proti oglišču in med vektorjem, ki poteka od središča lika proti referenčni točki za vrtenje. Na podlagi spremembe kota bomo vrteli lik
• Metoda, ki bo preverila, ali je treba dva lika zlepiti
• Metoda, ki bo preverila, ali sta si dva oglišča dovolj blizu
• Metoda, ki bo preverila, ali se oglišča dveh likov prekrivajo
• Metoda, ki bo zlepila dva lika skupaj
Ker bo lahko v seznamu poljubno število likov, bo seznam narejen s pomočjo povezanih podatkovnih struktur [24] – to smo predvideli že pri načrtovanju razreda Lik, ki vsebuje kazalec na naslednji element.
30
Poglejmo si še izvedbo nekaterih zgoraj naštetih metod:
• Metoda, ki preveri, ali smo pritisnili lik in vrne ID
public int getPressedShapeID(int x, int y) {
if (first != null) // Če seznam ni prazen, iščemo po likih {
Lik runner = first;
while (runner != null)
{ // preverimo, ali je točka (x,y) znotraj lika if (runner.isPointInside(x, y) )
{ // če je točka znotraj lika, označimo, da ga držimo runner.isPressed = true;
if (hasRotationReference) // če je vrtenje likov v teku { // izračunamo podatke za vrtenje še za ta lik
addRotationReference(runner);
}
return runner.id; // vrnemo id lika }
runner = runner.next; // pomik na naslednji objekt v seznamu }
}
return -1; // če točka (x,y) ni znotraj nobenega lika, vrnemo -1 }
• Metoda za dodajanje lika v seznam
Ta metoda je preprosta, treba pa je razumeti delo s kazalci.
public int addLik(Lik lik) {
lik.id = nextId; // liku določimo ID nextId++;
if (first == null) { first = lik;
last = lik;
} else {
last.next = lik;
last = last.next;
}
count++;
return lik.id; // vrnemo id lika }
31
• Metoda, ki preveri, kateri liki so prilepljeni k danemu liku in jih označi za premikanje
/**
* Preveri, če je kateri lik prilepljen (imata 2 skupna oglišča) in ga označi za
* premikanje. Rekurzivna metoda.
* @param shapeID - h kateremu liku ga prilepimo
* @param parentID - starš ki je sprožil preverjanje (kliknjen=
* @param excludeID - ID lika, ki ga želimo izključiti */
public void checkForGluedShapes(int shapeID, int parentID, int excludeID) {
if (first != null) {
Lik runner = first;
while (runner != null)
{ // lik, ki ga drži drug dotik izključimo
if (!runner.isPressed && (runner.id != excludeID) ) { // če imata lika dva skupna oglišča, sta zlepljena
if (getCommonVertexNumber(findShapeById(shapeID), runner) == 2)
{ // sproti dodajamo like v tabelo, ki jo
// bomo uporabili za preverjanje sestavljenih mrež if ( arrayItems < likArray.length )
{
likArray[arrayItems] = runner;
arrayItems++;
}
// označimo, da se premika in s katerim likom se // mora premikati
runner.willMove = true;
runner.gluedToID = parentID;
// ob ponovnem klicu metode trenutni lik //izključimo iz preverjanja, da ne pride do //neskončne zanke
checkForGluedShapes(runner.id, parentID, shapeID);
} }
runner = runner.next;
} }
}
Zgornja metoda je rekurzivna. Uporaba rekurzivne metode je najbolj primerna za problem iskanja sosedov. V tej metodi tudi pripravimo tabelo zlepljenih likov, katero bomo uporabili preverjanje, ali zlepljeni liki predstavljajo mrežo telesa.
• Metoda, ki premakne lik skupaj z liki, ki so zlepljeni
Ta metoda premika lik z danim id, poleg tega pa premakne še vse like, ki so k liku z danim id prilepljeni. Metoda prejme tri parametre. Prvi parameter je id lika, ki ga želimo premakniti, druga dva parametra sta pa razdalja premika v x in y smeri. V tej metodi se preverja tudi, ali smo lik približali dovolj blizu drugemu liku, da se zlepita.