Nic nie daje takiego natchnienia do pracy jak zbliżający się termin 🙂 Do końca miesiąca został tylko tydzień a styczbiowa gra nie jest jeszcze gotowa. Na szczęście nie zostało zbyt wiele pracy. W tej aktualizacji dodałem sporo nowości.
W lesie pojawiają się teraz potwory, które przeszkadzają wiedźmie w zbieraniu ziół. Drugą rzeczą mając na celu utrudnić czarownicy życie, są drzewa i kamienie, które blokują jej lot i muszą być omijane
W aktualną wersję gry pograć można klikając w obrazek powyżej. Jak zwykle jest też paczka z kodem i grafiką gry.
Tym razem naprawdę nie będę przedstawiał każdej nowej linijki kodu. Było by trudno, bo łącznie gra ma już ich prawie 900 a tej aktualizacji doszło ich około 300 🙂
Pierwszą duża zmianą jest pojawienie się w grze potworów. Podczas inicjalizacji głównego stanu gry, tworzę nowy obiekt creep. Będzie on wzorem, na podstawie którego powstawać będą potwory:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
this.creep = Object.create(self.spriteObject); this.creep.sourceY = 1636; this.creep.spawning = true; this.creep.spawnTime = 40; this.creep.spawningFor = 0; this.creep.type = undefined; this.creep.shooter = false; this.creep.frames = 11; this.creep.currFrame = 0; this.creep.dispTime = 2; this.creep.dispFor = 0; this.creep.speed = 6; this.creep.shotDown = false; this.creep.shootRate = 135; this.creep.lastShoot = 0; |
Zanim do gry dodany zostanie nowy potwór, w miejscu, którym ma się pojawić, wyświetla się animacja bąbelków. Animacja ta ma 11 klatek i trwa około 40 wywołań requestAnimationFrame. W trakcie pojawiania się wartość pola spawning potwora równe jest true. Za ten mechanizm odpowiada pierwsze parę pola w klasie creep.
Kolejne pola definiują czy potwór strzela i jakiego jest typu (czaszka, duch, nietoperz, oko). Są też pola obsługujące wyświetlanie animacji potwora. Definujące jego prędkość oraz fakt jak często może strzelać. Ważnym jest jeszcze pole shotDown, oznaczające czy potwór został zestrzelony. Potrzebuję tego dlatego, że potwór nie znika od razu po zestrzeleniu, w miejscu gdzie był znów pojawia się animacja bąbelków.
W funkcji update, doszła się spora ilość kodu obsługująca zachowanie potworów.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 |
//check if its creep spawn time if(this.creeps.length < 15) { if(this.lastCreep >= this.creepSpawnRate){ this.spawnCreep(); this.lastCreep = 0; } else { this.lastCreep++; } } //updateCreeps for(var i = 0; i<this.creeps.length;i++){ var creep = this.creeps[i]; if(creep.dispFor > creep.dispTime){ creep.currFrame++; if(creep.currFrame >= creep.frames){ creep.currFrame = 0; } creep.dispFor = 0; } else { creep.dispFor++; } if(creep.x<this.screen.x-100 || creep.x>this.screen.x+this.screen.width+100){ this.creeps.splice(i,1); } if(creep.spawning && creep.spawningFor >= creep.spawnTime){ creep.spawning = false; this.chooseCreep(creep,this.screen); } else { creep.spawningFor++; } if(!creep.spawning){ creep.update(); } if(creep.shooter && creep.lastShoot >= creep.shootRate){ this.castEvilMissile(creep,this.witch); creep.lastShoot = 0; } else { creep.lastShoot++; } if(creep.shotDown && creep.type !== "dead"){ creep.type = "dead"; creep.speed = 0; creep.sourceY = 1636; creep.sourceX = 384; creep.currFrame = 6; creep.frames = 12; } if(creep.type === "dead" && creep.currFrame === 10){ this.creeps.splice(i,1) } }; |
Najpierw sprawdzam czy w tablicy creeps, czyli tablicy przechowującej wszystkie potwory, które są aktualnie w grze, jest mniej niż 15 elementów. Jeśli tak i minął czas określający z jaką częstotliwością rodzą się potwory. Wywoływana jest metoda spawnCreep. Jest też resetowany licznik, odliczający czas do kolejnego potwora.
Od tego miejsca zaczyna się pętla for, która wykonuje całą resztę kodu dla każdego elementu w tablicy creeps
Kolejne parę linijek to oczywiście kod obsługujący animacje. Na początku, każdy potwór wyświetla animacje bąbelków. Kolejne wyrażenie if sprawdza czy potwór nie wyleciał poza granice ekranu plus sto pikseli. Jeżeli tak, to jest usuwany z tablicy creeps.
Dalej znajduje się kod, który sprawdza czy potwór wciąż się rodzi. Jeżeli czas potrzebny aby potwór się narodził minie, jego pole spawning ustawiane jest na false i wywoływana jest metoda chooseCreep.
Kolejne wyrażenie warunkowe, sprawdza czy aktualny potwór przestał już się pojawiać. Jeżeli tak, wywoływana jest jego metoda update, jest ona inna dla każdego typu potwora.
Następnie sprawdzane jest czy potwór strzela i czy minął wymagany czas od ostatniego strzału, jeżeli te warunki są spełnione, wywoływana jest metoda castEvilMissile.
Ostatnie dwa warunki obsługują sytuacje, w której potwór został zestrzelony. Jeżeli pole shotDown potwora równe jest true, ale jego typ, nie jest równy dead. Potwór przechodzi w stan wybuchu. Jego type otrzymuje wartość dead, a jego obrazek ustawiany jest na 6 klatkę bąbelków. Jeżeli potwór ma już typ równy dead i jest aktualnie na dziesiątej klatce, oznacza to, że minęła animacja pękającej bańki i potwór usuwany jest z tablicy creeps.
Jeśli chodzi o metody spawnCreep oraz chooseCreep, ich kod wygląda następująco:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 |
spawnCreep: function(){ var creep = Object.create(this.creep); creep.x = Math.floor(Math.random()*((this.screen.x+this.screen.width)-this.screen.x)+this.screen.x); creep.y = Math.floor(Math.random()*(self.canvas.height-150)); this.creeps.push(creep); }, chooseCreep: function(creep,screen){ var randomCreepNum = Math.floor(Math.random()*4); switch(randomCreepNum){ case 0: creep.type = "demonSkull"; creep.frames = 10; creep.sourceY = 1380; creep.shooter = true; creep.dispTime = 4; creep.update = function() { this.x += this.speed; if(this.x+this.width >= screen.x+screen.width){ this.speed = -this.speed; } if(this.speed < 0 && this.x <= screen.x) { this.speed = -this.speed; } }; break; case 1: creep.type = "ghost"; creep.frames = 4; creep.sourceY = 1444; creep.dispTime = 4; creep.update = function() { this.y += this.speed; if(this.y+this.height >= screen.height){ this.speed = -this.speed; } if(this.speed < 0 && this.y <= 0) { this.speed = -this.speed; } }; break; case 2: creep.type = "hellBat"; creep.frames = 4; creep.sourceY = 1508; creep.baseY = creep.y; creep.angle = 0; creep.waveRange = 150; creep.dispTime = 4; creep.update = function() { this.x += this.speed; if(this.x+this.width >= screen.x+screen.width){ this.speed = -this.speed; } if(this.speed < 0 && this.x <= screen.x) { this.speed = -this.speed; } this.y = this.baseY + Math.sin(this.angle) * this.waveRange; this.angle += 0.01; if(this.angle > Math.PI*2){ this.angle = 0; }; if(this.y<=0){ this.y = 0; } if(this.y+this.height>=screen.height){ this.y=screen.height; } }; break; case 3: creep.type = "evilEye"; creep.frames = 4; creep.dispTime = 8; creep.sourceY = 1572; creep.shooter = true; creep.update = function() { }; break; } }, |
spawnCreep, jest proste. Tworzony jest nowy obiekt potworka na podstawie obiektu creep. Następnie, losowo wybierane są współrzędne dla niego. Przygotowany potwór jest wrzucany do tablicy creeps. Należy pamiętać, że na początku potworek to tylko bąbelki, dopiero kiedy skończy się rodzić wywoływana jest funkcja chooseCreep.
W chooseCreep losowo wybierany jest jeden z typów potwora. Za pomocą konstrukcji switch, obiekt otrzymuje odpowiednie dane co do nowego obrazka, liczby klatek, czasu wyświetlania, typu potworka oraz czy jest on strzelający czy nie. Do tego, każdy typ ma swoją funkcję update. Ta funkcja inna dla każdego rodzaju. Jej celem jest, znaleźć położenie potwora w następnej klatce. Dlatego niektóre potwory poruszają się w poziomie, inne w pionie a jeszcze inne po łukach. Jeden potworek nie porusza się wcale :).
Ostatnia rzecz tycząca się bezpośrednio potworków, to wypuszczane przez nie pociski. Są one tworzone na podstawie klasy evilMissile, która z kolei tworzona jest w metodzie init na podstawie magicMissile:
1 2 3 4 |
this.evilMissile = Object.create(this.magicMissile); this.evilMissile.sourceX = 60; this.evilMissile.angle = 0; this.evilMissile.speed = 6; |
Instancje tego obiektu tworzone są w metodzie castEvilMissile, która wywoływana jest w metodzie update, gdy tylko u któregoś licznik mierzący okres czasu między strzałami się wyzeruje
1 2 3 4 5 6 7 |
castEvilMissile: function(creep,witch){ var missile = Object.create(this.evilMissile); missile.x = creep.x+(creep.width/2); missile.y = creep.y+(creep.height/2); missile.angle = Math.atan2(witch.y-creep.y,witch.x-creep.x); this.evilMissiles.push(missile); }, |
Używam wbudowanej funkcji trygonometrycznej atan2, aby ustalić kąt pomiędzy potworem a wiedźmą. Dzięki tej informacji będę mógł spowodować, że pocisk będzie leciał dokładnie w miejsce w którym znajduje się wiedźma podczas jego wystrzału. Obliczam to w metodzie update
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 |
for(var i = 0; i < this.evilMissiles.length; i++){ var missile = this.evilMissiles[i]; if(missile.dispFor > missile.dispTime){ missile.currFrame++; if(missile.currFrame === missile.frames){ missile.currFrame = 0; } missile.dispFor = 0; } else { missile.dispFor++; } missile.x += Math.cos(missile.angle) * missile.speed; missile.y += Math.sin(missile.angle) * missile.speed; if(missile.x < this.screen.x-500 || missile.x > this.screen.x+this.screen.width+500){ this.evilMissiles.splice(i,1); } } |
Najpierw oczywiście aktualizowane są klatki animacji pocisku. Następnie używany jest kąt w funkcjach trygonometrycznych sin oraz cos. Dzięki nim, mogę wyliczyć położenie pocisku poruszającego się pod odpowiednim kątem. Nie będę teraz opisywał dokładnie jak to działa, poświęcę temu osobny post w niedalekiej przyszłości 🙂 . Przy okazji sprawdzam, też czy pocisk potwora nie wyleciał daleko poza ekran, jeżeli tak, usuwam go z tablicy.
Teraz pozostało już tylko wykrycie kolizji, pomiędzy czarownicą a stworami. Oczywiście dzieje się to w metodzie update:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 |
for(var i = 0;i<this.creeps.length;i++){ var creep = this.creeps[i]; if(!creep.spawning && !creep.shotDown && this.checkCollision(this.witch,creep)){ creep.shotDown = true; if(!this.witch.recovering){ this.witch.lifePoints -= 10; this.witch.recovering = true; } } } for(var i = 0;i<this.evilMissiles.length;i++){ var missile = this.evilMissiles[i]; if(this.checkCollision(this.witch,missile)){ this.evilMissiles.splice(i,1); if(!this.witch.recovering){ this.witch.lifePoints -= 10; this.witch.recovering = true; } } } |
W pierwszej pętli sprawdzam czy czarownica styka się z potworem, który nie rodzi się (tylko bąbleki) lub nie został już zestrzelony (już tylko bąbelki). Jeżeli tak, potworek zostaje oznaczony jako trafiony (shotDown). Do tego jeśli czarownica nie ma pola recovering równego true, traci 10 z wartości pola lifePoints (początkowo równe 100) i recovering ustawiane jest na true. Oznacza to, że po zderzeniu z potworem, czarownica traci punkty życia a potwór ginie.
To samo tyczy się pocisków potworów, jeśli nastąpi kolizja pomiędzy nimi a czarownicą, traci ona punkty życia jej pole recovering ustawiane jest na true, a pocisk jest usuwany z gry.
O co chodzi z recovering? Jest to mechanizm, który sprawia że przez pewien czas po otrzymaniu obrażeń, czarownica nie może otrzymać kolejnych. Obsługiwane jest to w metodzie update na podstawie trzech pól obiektu witch:
1 2 3 |
this.witch.recovering = false; this.witch.recoverTime = 90; this.witch.recoveredFor = 0; |
Jest tu oczywiście licznik, oraz czas jaki trwać będzie niewrażliwość czarownicy. Fragment metody update który to obsługuje wygląda następująco:
1 2 3 4 5 6 7 8 9 10 11 12 13 |
if(this.witch.recovering){ if(this.witch.recoveredFor >= this.witch.recoverTime){ this.witch.recovering = false; this.witch.sourceY = 600; this.witch.recoveredFor = 0; } else { if(this.witch.recoveredFor%10 === 0 && this.witch.sourceY === 600){ this.witch.sourceY = 9000000; } else if(this.witch.recoveredFor%10 === 0 && this.witch.sourceY !== 600){ this.witch.sourceY = 600; } this.witch.recoveredFor++; } |
Oprócz odliczania czasu, co 10 obiegów sprawiam, że zamiast na obrazek wiedzmy pole sourceY obiektu wskazuje na puste miejsce na arkuszu obrazków. Dzięki temu, po otrzymaniu obrażeń, czarownica miga.
Oczywiście w metodzie update nie zabrakło też wykrycia kolizji pomiędzy potworami a pociskami czarownicy:
1 2 3 4 5 6 7 8 9 10 |
for(var i = 0; i < this.magicMissiles.length; i++){ var missile = this.magicMissiles[i]; for(var j = 0; j < this.creeps.length; j++){ var creep = this.creeps[j]; if(!creep.spawning && this.checkCollision(missile,creep)){ this.magicMissiles.splice(i,1); creep.shotDown = true; } } } |
Każdy pocisk sprawdzany jest z każdym potworem. Jeżeli występuje kolizja, a potwór nie jest w trakcie rodzenia się. Potwór oznaczany jest jako ustrzelony, a pocisk usuwany jest z gry.
W ten bardzo ogólny sposób udało mi się opisać dodanie potworów czyli główną zmianę w grze i wszystkie jej następstwa 🙂
Kolejna trochę mniejsza zmiana, zamiana drzew i kamień na mapie w obiekty, które zatrzymują czarownicę. Nie są one już częścią obrazka tła jak wcześniej a są pełnoprawnymi obiektami dodawanymi w trakcie inicjacji gry. Tak wygląda ich kod:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
this.obstacle = Object.create(self.spriteObject); this.oak = Object.create(this.obstacle); this.oak.sourceY = 1700; this.oak.sourceWidth = 162; this.oak.sourceHeight = 212; this.oak.width = 70; this.oak.height = 212; this.fir = Object.create(this.obstacle); this.fir.sourceX = 162; this.fir.sourceY = 1700; this.fir.sourceWidth = 162; this.fir.sourceHeight = 212; this.fir.width = 70; this.fir.height = 212; this.stone = Object.create(this.obstacle); this.stone.sourceY = 1912; this.stone.sourceWidth = 71; this.stone.sourceHeight = 51; this.stone.width = 71; this.stone.height = 51; |
Jak widać, są to zwykłe obiekty ze zdefiniowanymi współrzędnymi i rozmiarami. W tym miejscu warto zwrócić uwagę, że rozmiar ich obrazków jest większy od faktycznego rozmiaru. To dlatego, że chcę aby czarownica mogła nalecieć na część obrazka drzewa (wlecieć w liście), ale zatrzymać się na jego środku (na pniu 🙂 ). Te obiekty definiowane są w metodzie init. W tej samej metodzie, zaraz po definicji obiektów, wywoływana jest metoda populateObstacles.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 |
populateObstacles: function(){ for(var i = 0;i<this.obstacleXs.length;i++){ var randomObstacle = Math.floor(Math.random()*3); var randomObstacleYmod = Math.floor(Math.random()*11-(-10)+(-10)); var obstacle; switch(randomObstacle){ case 0: obstacle = Object.create(this.oak); obstacle.x = Math.floor(this.obstacleXs[i]+((obstacle.sourceWidth-obstacle.width)/2)); obstacle.drawX = this.obstacleXs[i]; obstacle.y = 320 + randomObstacleYmod; break; case 1: obstacle = Object.create(this.fir); obstacle.x = Math.floor(this.obstacleXs[i]+((obstacle.sourceWidth-obstacle.width)/2)); obstacle.drawX = this.obstacleXs[i]; obstacle.y = 320 + randomObstacleYmod; break; case 2: obstacle = Object.create(this.stone); obstacle.x = this.obstacleXs[i]; obstacle.drawX = this.obstacleXs[i]; obstacle.y = 460 + randomObstacleYmod; break; } this.obstacles.push(obstacle); } }, |
W głównym obiekcie stanu zdefiniowałem tablicę, zawierającą współrzędne X każdego z obiektów. Metoda powyżej przechodzi przez tę tablicę i dla każdego elementu tworzy obiekt obstacle. Losowo wybierane jest czy będzie do dąb, sosna czy kamień. Losowo wybierana jest też jego współrzędna Y, chociaż tak, żeby położenie przeszkody miało sens. Dla drzew tworzę też pole drawX, wskazujące gdzie ma być narysowany obrazek. W ten sposób sam obiekt będzie umieszczony wewnątrz obrazka.
Teraz potrzebna jest tylko funkcja obsługująca kolizje pomiędzy czarownicą a przeszkodami. Tak wygląda fragment kodu metody update, który za to odpowiada:
1 2 3 |
for(var i = 0; i<this.obstacles.length; i++){ this.blockRectangle(this.witch,this.obstacles[i]); } |
Co każdy cykl gry, dla każdej przeszkody wywoływana jest metoda blockRectangle. Sprawdza ona kolizję pomiędzy daną przeszkodą a wiedźmą:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 |
blockRectangle: function(r1, r2) { var vx = r1.centerX() - r2.centerX(); var vy = r1.centerY() - r2.centerY(); var combinedHalfWidths = r1.width/2 + r2.width/2; var combinedHalfHeights = r1.height/2 + r2.height/2; if(Math.abs(vx) < combinedHalfWidths){ if(Math.abs(vy) < combinedHalfHeights){ var overlapX = combinedHalfWidths - Math.abs(vx); var overlapY = combinedHalfHeights - Math.abs(vy); if(overlapX >= overlapY) { if(vy > 0) { r1.y = r1.y + overlapY; } else { r1.y = r1.y - overlapY; } } else { if(vx > 0) { r1.x = r1.x + overlapX; } else { r1.x = r1.x - overlapX; } } } } }, |
Nie będę dokładnie opisywał działania tej metody. Zasługuje na osobny post, ponieważ daje bardzo dużo ciekawych możliwości. Opowiem tylko ogólnie jak działa. Najpierw pomiędzy dwoma prostokątnymi obiektami wykrywana jest kolizja. Jeżeli zachodzi, metoda sprawdza z której strony obiekty się stykają. Następnie drugi obiekt, cofany jest w tę stronę o tyle pikseli o ile obiekty się pokrywają. W ten sposób pierwszy obiekt, blokuje drugi.
I to wszystkie nowości. Tym razem nie opisywałem każdej nowej zmiennej. Większość mechanizmów się powtarza. Jak zwykle jeśli coś nie jest jasne, pytaj w komentarzu. Z chęcią odpowiem na każde pytanie.
Myślę że gra zaczyna nabierać bardzo konkretnych kształtów. Kolejny krok to dodanie GUI oraz zmiana stanów pomiędzy główną grą tak aby były bardziej atrakcyjne. I to będzie już cała gra. Ostatniej aktualizacji można spodziewać się pod koniec tego tygodnia. Jeżeli chcesz być pewny, że jej nie przegapisz, zachęcam do polubienia mojej strony na facebooku. Zamieszczam tam wszystkie informacje o nowościach na bieżąco 🙂