Programmera spel i C++ för nybörjare/Exempel på collision detect i spel
Denna genomgång förutsätter att du har en fungerande stridsvagnskod från exemplet "Programmera spel i C++ för nybörjare/Enkelt stridsvagnspel" Jag använder koden för Vectorer och inte för Array, eftersom det ger bäst minneshantering och oändligt antal kulor både för stridsvagnen och för kanonen.
Det är väl ok att kunna skjuta med sin spelare, men hur vet man att man får en träff på en motspelare? Genom att bygga ut koden och lägga till en kanon kan vi få ett spel där en spelare kan köra omkring med en stridsvagn och en annan spelare har en stillastående kanon och försöker att träffa stridsvagnen. Genom detta kan vi se hur de tre vanligaste varianterna av collision detect fungerar i praktiken. Du kan läsa mer om teorierna bakom collision detect här: Programmera spel i C++ för nybörjare/Collision detect
Grafik
redigeraNy grafik är bilden av en kanon du kan ladda hem här:
http://strategywiki.org/wiki/Assault/Getting_Started
Jag valde bilden: AT_Type_1_Cannon.gif från listan i botten av sidan. Bilden måste roteras 180 grader så att kanonpipan pekar uppåt och därefter sparas om i png-format. Originalets gif-format fungerar inte i SFML. Glöm inte att ange att den svarta färgen i bilden skall vara genomskinlig.
kanonen
redigeraVi vill ha ut kanonen på spelplanen så att vi kan börja skjuta på den och kontrollera de olika sätten man kan se om man fått en träff på. Egentligen skulle vi så klart ha den här kanonen som en ny klass som ärver från fordon, men eftersom jag bara vill visa hur det går till när två enheter skjuter på varandra låter jag kanonen vara en ny pansarvagn fast med hastighet 0 och bensin 0. Den placeras ut i nedre högra hörnet på koordinaten x=700, och y=500 eftersom spelplanen är 800 x 600 stor. Den får mindre mängd pansar också.
//stridsvagn(int ut_hastighet, double ut_spelare_x, double ut_spelare_y, int ut_bensin, int ut_pansar) stridsvagn kanon1(0.0f, 700.0,500.0,0,50);
Vi laddar in grafiken för kanonen i spelet:
//Skapa bild för kanonen sf::Image kanonbild; //skapa en tom bildhållare som heter bild if (!kanonbild.LoadFromFile("AT_Type_1_Cannon.png")) return EXIT_FAILURE; //fyll den tomma bildhållaren med bilden på kanonen kanon1.sprite.SetImage(kanonbild);
Vi placerar ut kanonen på spelplanen:
//Placera också ut kanonen på spelplanen kanon1.sprite.SetPosition(kanon1.spelare_x,kanon1.spelare_y);
Slutligen vill vi kunna rotera den runt sin egen axel istället för runt övre vänstra hörnet:
kanon1.sprite.SetCenter(kanon1.sprite.GetSize().x / 2 , kanon1.sprite.GetSize().y / 2);
Vid rutinen för att rita upp grafik på skärmen lägger jag till:
App.Draw(kanon1.sprite); //Rita upp kanonen
Det är vad som behövs för att få fram kanonen på spelplanen så att vi kan börja övningsskjuta på den.
Rotation, sikta med kanonen
redigeraNaturligtvis vill vi kunna sikta med kanonen också, men det finns bara en mus till datorn. Därför kopplar vi rotationskoden för kanonen till piltangenterna istället:
//Rotera kanonen med höger/vänster piltangent if (App.GetInput().IsKeyDown(sf::Key::Right)) kanon1.sprite.Rotate(- 10); //medsols if (App.GetInput().IsKeyDown(sf::Key::Left)) kanon1.sprite.Rotate(+ 10); //motsols
Man skulle naturligtvis lika gärna kunna skriva:
if ((Event.Type == sf::Event::KeyPressed) && (Event.Key.Code == sf::Key::Right)) kanon1.sprite.Rotate(- 10);//medsols if ((Event.Type == sf::Event::KeyPressed) && (Event.Key.Code == sf::Key::Left)) kanon1.sprite.Rotate(+ 10); //motsols
Resultatet blir exakt likadant, men det blir mer kod och den här koden måste ligga inom
while (App.GetEvent(Event)) {...}
Hur du gör är mer litet av en vana (eller ovana).
Första varianten: Cirkeltest
redigeraPrincipen bakom cirkeltest är väldigt enkel och till spel där spelarna är runda eller kvadratiska duger den här typen av kontroll om man har träffat.
- Ta reda på avståndet mellan mittpunkterna mellan de två spelpjäserna
- Vilken radie har den första spelpjäsens bild?
- Vilken radie har den andra spelpjäsens bild?
- Om radien för den första + radien för den andra är större än avståndet mellan dem, har ingen träff skett.
- Om radien för den första + radien för den andra är lika med- eller mindre än avståndet mellan dem, har en träff skett.
Pythagoras sats
redigeraKommer du ihåg pythagoras sats från grundskolan? I en rätvinklig triangel (en vinkel i triangeln är 90 grader) är den längsta sidans (hypotenusans) längd samma som kvadraten ur basens längd ^2 + höjdens längd ^2 (^ tecknet = "upphöjt till" vid datorprogrammering). I en spelvärld kommer alltid en vinkel att vara 90 grader mellan två spelare med två undantag. Om båda spelpjäsernas x koordinat är likadan, eller båda spelpjäsernas y koordinat är likadan. Då står nämligen bägge spelpjäserna på samma linje och en triangel kan inte bildas.
Anta att vi vill få fram avståndet mellan mittpunkterna i stridsvagnen och kanonen vid spelstarten.
- Stridsvagnens X= 100 och Y= 100
- Kanonens X= 700 och Y= 500.
- Skillnaden i Y: 500 - 100 = 400.
- Skillnad i X: 700 - 100 = 600.
- Avståndet mellan punkterna ^2 = 600^2 + 400 ^2.
600 ^2 (samma som 600 * 600)= 360 000 400 ^2 (samma som 400 * 400)= 160 000
- Kvadratroten (tecknet ser ut som ett v med en extra horisontell linje på miniräknaren) ur 360 000 + 160 000 = 721.11
Då vet vi avståndet mellan mittpunkterna. Då måste vi ta reda på radien av figurerna. Bägge är 64 x 64 så det är lätt att tänka sig att radien är halva den summan = 32, men så enkelt är det inte om spelpjäserna är fyrkantige. Då täcks inte hörnen in och en kula som ser ut som om den träffade ett hörn skulle bara skjutas förbi.
- Radien måste beräknas på avståndet mellan övre vänstra hörnet och nedre högra hörnet delat med två (annars får du diametern) om spelpjäsen är fyrkantig. Om det är en rund pjäs (t.ex. en biljardboll) kan man låta diametern vara samma som avståndet mellan två sidor.
- Diametern är då kvadratroten ur 64^2 + 64^2 ungefär = 90.4
- Radien är halva den summan = 45.2
- Slår man ihop radien för de två spelpjäserna ser man att så länge som avståndet mellan deras mittpunkter är större än 90.4 kommer de inte att krocka.
Negativa tal
redigeraVad händer om talet blir negativt? Spelare ett befinner sig till höger om spelare två och x blir 100 - 700 = -600? Inga problem. Två negativa tal som multipliceras med varandra blir alltid ett positivt tal. -600 * -600 är samma sak som -600^2 och resultatet blir alltså detsamma som 600 * 600 eller 600^2.
x1 = x2 eller y1 = y2
redigeraOm spelare 1 x koordinat är samma som spelare 2 x koordinat innebär det att de befinner sig lika långt in från kanten men på olika höjd. De befinner sig i vertikal linje med varandra. Då blir avståndet mellan deras mittpunkter lika med spelare 1 y koordinat - spelare 2 y koordinat.
Om spelare 1 y koordinat är samma som spelare 2 y koordinat innebär det att de befinner sig lika långt in från överkanten men på olika avstånd från vänsterkanten. De befinner sig i horisontell linje med varandra. Då blir avståndet mellan deras mittpunkter lika med spelare 1 x koordinat - spelare 2 x koordinat.
Oftast är det bäst att kontrollera om de ligger i linje med varandra innan man beräknar pythagoras sats eftersom det är en mycket enklare beräkning för datorn att göra. Då går spelet en gnutta snabbare.
Kvadratiska spelpjäser
redigeraOm du bara har kvadratiska spelpjäser är det lättare att räkna ut diametern. Det finns nämligen en konstant för det: 1.4142135623730950488016887242097
- Om varje sida är en pixel blir diametern (hypotenusan) 1.4142135623730950488016887242097
- Om varje sida har 64 bitar blir diametern (hypotenusan) 1.4142135623730950488016887242097 * 64 = 90.4 och radien halva den summan = 45.2
Du gör det alltså enkelt för dig om så många spelpjäser som möjligt är kvadratiska och du definierar konstanten i början. Då spar du en fruktansvärd mängd flyttalsberäkningar för datorn och prestandan på spelet ökas.
Träff?
redigeraDe tidigare koden vi använde visade en explosion om kulan var 50 pixels eller närmare kanten. Då bildades en explosion. Funktionen var väldigt enkel och såg ut så här:
//Kolla om kulan har träffat en vägg*********************************************************** bool check_hit(double x, double y) //se om kulan träffat { bool kulan_har_traeffat = false; if (x < 50 || x > 750) kulan_har_traeffat = true; if (y < 50 || y > 550) kulan_har_traeffat = true; return kulan_har_traeffat; }
Nu får vi lov att ändra funtionen totalt. Vi måste ha med x och y koordinaterna för två olika sprites. Endera skickar vi in fyra olika double-värden, eller så skickar vi in två olika adresser till spritesen. För att göra det tydligt kallar vi den här funktionen Cirkular (så att vi får korrekt namn på de tre olika sätten som finns).
//Kolla om kulan har träffat en annan spelare*********************************************************** bool Cirkular_check_hit(sf::Sprite &kulsprite, sf::Sprite &maolsprite) //se om kulan träffat { bool kulan_har_traeffat = false; //returvärde double avstaond = 0.0; //Avståndet mellan två sprites mittpunkter double traeffavstaond = 0.0; //sammanlagda summan av två sprites radie double tempx; //sträckan i x-led double tempy; //sträckan i y-led //Ta fram koordinaterna för de två spritesen. //kulax och kulay är kanonkulans koordinater //maolx och maoly är måltavlans koordinater double kulax = kulsprite.GetPosition().x; double kulay = kulsprite.GetPosition().y; double maolx = maolsprite.GetPosition().x; double maoly = maolsprite.GetPosition().y; //Ta fram kulansradie, 16 x 16 pixels stor double kula_radie = (1.4142135623730950488016887242097 * 16) / 2; //Ta fram målets radie, 64 x 64 pixels stor double maol_radie = (1.4142135623730950488016887242097 * 32) / 2; //Så när träffar dem? vilket avstånd minst från varandra? traeffavstaond = kula_radie + maol_radie; if (kulax == maolx && kulay != maoly ) //de står på samma linje i x-led { avstaond = kulay - maoly; if (avstaond <= traeffavstaond) //inom träffavstånd kulan_har_traeffat = true; } else if (kulay == maoly && kulax != maolx ) //de står på samma linje i y-led { avstaond = kulax - maolx; if (avstaond <= traeffavstaond) //inom träffavstånd kulan_har_traeffat = true; } else if(kulax != maolx && kulay != maoly) //De står på olika platser både x och y { //Räkna med pythagoras tempx = kulax-maolx; tempy = kulay-maoly; // Räkna ut hypotenusan/avståndet genom att ta // kvadratroten ur ena sträckan ^2 + andra sträckan ^2 avstaond = sqrt ( tempx * tempx + tempy * tempy); if (avstaond <= traeffavstaond) //inom träffavstånd kulan_har_traeffat = true; } //Räkna med pythagoras else if (kulax == maolx && kulay == maoly) //De står på exakt samma plats kulan_har_traeffat = true; return kulan_har_traeffat; }
Lägg in funktionsdeklarationen alldeles före main börjar (glöm inte semikolon i slutet):
bool Cirkular_check_hit(sf::Sprite &kulsprite, sf::Sprite &maolsprite); //se om kulan träffat
Slutligen kommenterar vi bort den tidigare testen för en träff och istället skriver vi:
//Kolla träff om kulan är 50 pixlar från kanten // if (check_hit(vSkottlista.at(si).Sprite.GetPosition().x, vSkottlista.at(si).Sprite.GetPosition().y) == true) //kolla träff för cirkular hit mellan kulan och kanonen if ( Cirkular_check_hit(vSkottlista.at(si).Sprite, kanon1.sprite)== true)
Det är allt. När du nu kör omkring och skuter med stridsvagnen ser du att du kan skjuta iväg kulorna utanför kanten, men så fort kulan tar i kanten på kanonen får du en explosion just där.
Andra varianten: Bounding box
redigeraAlla sprites och spelpjäser är inte runda eller kvadratiska. Många är avlånga. Exempelvis en torped som skickas från en ubåt i ett undervattensspel. Att utgå ifrån diametern på något som t.ex. är 10 pixlar brett men 80 pixlar långt gör att man kommer att få mängder med fel. Skjuter man mot torpeden från sidan kommer explosionen att uppstå 40 pixlar ifrån mitten trots att den bara är 10 pixlar bred.
Ett sätt att komma runt det är att istället tänka sig att torpeden ligger inuti en fyrkant och att kulan som skjuter mot torpeden inte är rund utan också ligger inuti en fyrkant. sedan kontrollerar man om fyrkanterna kommit innanför varandra. Det är enklare än det låter, i alla fall i spel där man bara använder sig av fyrkanter med 90 graders vinklar i hörnen. det är oändligt mycket krångligare i 3D...
Pseudokod är enligt följande:
- Är kulan rakt ovanför torpeden och har kulans nedre vägg passerat torpedens övre vägg?
- Är kulan rakt under torpeden och har kulans övre vägg passerat torpedens nedre vägg?
- Är kulan i nivå med torpeden och har kulans högra vägg passerat torpedens vänstra vägg?
- Är kulan i nivå med torpeden och har kulans vänstra vägg passerat torpedens högra vägg?
- Om någon av ovanstående frågor besvarats med ja har en träff skett, annars har ingen träff skett.
- Eller; om någon del av kulan är under torpedens överkant och över torpedens underkant och till höger om torpedens vänsterkant och till vänster om torpedens högerkant, har en träff skett.
För att det här skall fungera måste vi veta två x och två y värden för båda spelpjäserna som skall testas eller också utgår vi från de två spelpjäsernas mittpunkter (vilket är enklast) och beräknar var kanterna finns utifrån mittpunktens position.
Nu döper vi funktionen till Box_check_hit för att komma ihåg att vi kontrollerar om vi fått en träff genom att box krockar med box:
//Kolla om kulan har träffat kanten av en annan spelare*********************************************************** bool Box_check_hit(sf::Sprite &kulsprite, sf::Sprite &maolsprite) //se om kulan träffat { bool kulan_har_traeffat = false; //returvärde //Ta fram koordinaterna för de två spritesen. //kulax och kulay är kanonkulans koordinater //maolx och maoly är måltavlans koordinater double kulax = kulsprite.GetPosition().x; double kulay = kulsprite.GetPosition().y; double maolx = maolsprite.GetPosition().x; double maoly = maolsprite.GetPosition().y; //Kulan är 16 x 16 och målet är 64 x 64, vi räknar från varje sprites mittpunkt // innanför vänster kant innanför höger kant innanför över kant innanför neder kant if ((kulax + 8 > (maolx - 32)) && (kulax - 8 < maolx + 32) && kulay + 8 > maoly -32 && kulay - 8 < maoly + 32 ) kulan_har_traeffat = true; //returvärde return kulan_har_traeffat; }
Lägg in funktionsdeklarationen före main (glöm inte semikolon):
bool Box_check_hit(sf::Sprite &kulsprite, sf::Sprite &maolsprite); //se om kulan träffat boxkan
Lägg slutligen in funktionen genom att kommentera bort föregående funktion:
//if ( Cirkular_check_hit(vSkottlista.at(si).Sprite, kanon1.sprite)== true) if (Box_check_hit(vSkottlista.at(si).Sprite, kanon1.sprite) == true)//se om kulan träffat
När du skjuter nu kan du försöka att träffa kanonens hörn. Eftersom kanonen har en rätt rund form, och även kanonkulan är rund, är det ganska tydligt att kulan träffar alldeles innan den egentligen tar i kanonen, vilket är största problemet när man använder bounding box tester på just runda föremål.
Generell "bounding box" funktion
redigeraFunktionen här ovanför är byggd för just det här spelet med just de spelpjäsernas storlekar. Vad händer om man gör ett annat spel med andra spelpjäser? Här följer en mer avancerad funktion men som kan användas för att snabbt se om en kollision skett oavsett storleken på boxarna. Om man sätter koordinatpunkten på något annat ställe än i övre vänstra hörnet för en sprite, måste koden justeras om:
short int Sprite_Collide(sf::Sprite &spelare, sf::Sprite &fiende) { //alla "1" är spelaren och "2" är fienden //Koden bygger på att man har koordinaterna i övre vänstra hörnet //Om man vill utgå ifrån en träff där man beräknar från mittpunkter får man dessutom skriva // spelare.SetCenter(spelare.GetSize().x / 2 , spelare.GetSize().y / 2); // fiende.SetCenter(fiende.GetSize().x / 2 , fiende.GetSize().y / 2); float left1, left2; //Boxarnas vänsterkanter float right1, right2; //Boxarnas högerkanter float top1, top2; //Boxarnas överkanter float bottom1, bottom2; //Boxarnas underkanter left1 = spelare.GetPosition().x; //spelarens-x koordinat left2 = fiende.GetPosition().x; //målets x-koordinat right1 = spelare.GetPosition().x + spelare.GetSize().x; //spelarens högerkant right2 = fiende.GetPosition().x + fiende.GetSize().x; //måltavlans högerkant top1 = spelare.GetPosition().y; //spelarens överkant top2 = fiende.GetPosition().y; //målets överkant bottom1 = spelare.GetPosition().y + spelare.GetSize().y; //spelarens underkant bottom2 = fiende.GetPosition().y + fiende.GetSize().y; //målets underkant if (bottom1 < top2) return(0); // Spelaren flyger över målet if (top1 > bottom2) return(0); // Spelaren flyger under målet if (right1 < left2) return(0); // Spelaren är bortanför målet åt höger if (left1 > right2) return(0); // Spelaren är bortanför målet åt vänster return(1); //Träff! Den har skett om inte spelaren är bortanför målet //Originalkod från: http://www.gamedev.net/page/resources/_/technical/game-programming/collision-detection-r735 };
För att avgöra om det blivit en träff skriver du således bara:
int Sprite_Collide = 0; if (Sprite_Collide(spelare.sprite, fiende.sprite) == 1) {//Träff, gör något}
Kvicksand...
redigeraAnta att du har en gravitation som pressar neråt och du, fast du står på en plattform du programmeringsmässigt fått en träff mot enligt koden här ovanför, ändå sjunker neråt långsamt som om spriten fastnat i kvicksand, har du problem. Du har en del av din fyrkant innanför plattformens "bounding box" (om det är ett plattformsspel) vilket indikerar träff hela tiden och du sjunker långsamt, långsamt genom plattformen. För att undvika det får du se till så att din sprite befinner sig alldeles, alldeles utanför boxen, 0.1 pixel fungerar, eller på exakt samma linje.
Var träffar man?
redigeraNu vet du att du fått en träff, men var har du träffat? Det finns egentligen tre olika sätt att lista ut det:
- Jämför koordinaterna för mittpunterna för både spelaren och motståndaren. Är spelaren ovanför och krockar t.ex. kan vi anta att spelarens undersida krockat med motståndarens översida.
- Utgå ifrån din hastighet i x och y. Om spelaren färdas med negativt x (åt vänster) kan man anta att man krockat med sin egen vänstersida mot en motståndares högersida.
- Svårast, utgå ifrån vinkeln mellan spelaren och motståndaren. Är vinkeln 0 grader är man rakt ovanför, 90 grader rakt åt vänster, 180 grader rakt ner osv. Vinkeln räknas lättast ut med cos eller sinus när man vet hypotenusans och kateternas längd.
Tredje varianten: Pixel perfect
redigeraDen tredje sortens kontroll om man träffat är tung för datorn att beräkna. Därför används den vanligtvis inte ensam utan tillsammans med någon av de två övriga sätten. Först kontrollerar man om man fått en träff. Sedan gör man en pixel perfect kontroll för att se om det verkligen var en träff. Det är ett bra sätt för att t.ex. undvika att man får explosioner vid hörnen på runda saker om man gör en bounding box test. Så vad innebär egentligen pixel perfect? Jo, man har alltid en färg som är genomskinlig, men det är faktiskt en färg. I just vårt exempel har pansarvagnen mörkgrönt, kulan har nästan svart och kanonen har svart färg som markerats som genomskinliga. Efter det att man fått en "träff" enligt de andra teknikerna kontrollerar man varenda pixel under kanonkulan (i det här fallet) och ser vilken färg den har. Om samma pixelkoordinat i förhållande till spelplanen är genomskinlig både på kanonen och på kanonkulan, eller på någon av spelarna har ingen träff skett men om båda pixlarna inom den överlappande arean inte är genomskinliga kan man anta att det skett en träff. Krångligt? Javisst. Knappast för nybörjare heller, men jag vill i alla fall nämna det.
Tänk efter före
redigeraOm du tänker använda dig av pixel perfect i stor utsträckning är det en god idé att se till så att samtliga sprites har exakt samma färg som är genomskinlig. På det viset underlättar du kodningen rejält för dig själv. Om du har sprites som inte skall stoppa kulor, som t.ex. rök, kan du ge den en annan genomskinlig bakgrundsfärg. Genom att ge alla enheter som kan träffas av kulor samma genomskinliga bakgrundsfärg kan du använda samma funktion till samtliga spelpjäser. Något som underlättar din kodning rejält.
Genomskinliga färger
redigeraNär vi startar kodningen är det bra att veta vilka färger som anses genomskinliga i spelet. I vårt lilla spel är det t.ex:
- Genomskinlig färg på kulan = 1,0,0 (RGB)
- Genomskinlig färg på kanonen = 0,0,0 (RGB)
- Genomskinlig färg på stridsvagnen = 59,147,81 (RGB)
Man använder sig sällan av pixel perfect. Det tar lång tid att gå alla dessa pixlar på alla berörda spelpjäser eftersom man måste göra flera olika beräkningar för att se om man fått en träff. Pseudokoden kan se ut så här:
- Vilka koordinater har fyrkanten som bildas när två spelpjäser överlappar varandra?
- Vilka av de koordinaterna finns i både kanonkulan och i målet (i vårt exempelfall)?
- Har dessa koordinater genomskinliga färger på bägge spelpjäserna på samma plats?
- Är alfakanalen ställd till 0? Då kan du sluta försöka eftersom genomskinligheten också ställts till 0.
Vill du ändå fördjupa dig finns dels ett exempel på SFML:s hemsida och dels en riktigt bra artikel här:
http://www.gamedev.net/page/resources/_/technical/game-programming/collision-detection-r735
Färg för kollisionsberäkningar
redigeraOm du behöver ta fram färgen på en exakt punkt i en image som en sprite har gör du så här:
//Definiera färgerna: int red = 255; int green = 255; int blue = 255; //koordinater int x_koordinatibilden=0; int y_koordinatibilden=0; //Inuti main const sf::Color color = Sprite.GetPixel (x_koordinatibilden, y_koordinatibilden); red = (int)color.r; green = (int)color.g; blue = (int)color.b;
Därefter får du jämföra pixelns färg och kontrollera om den överensstämmer med RGB värdet för den genomskinliga färgen. Tänk på att om du använder ett spritesheet får du fram hela sprite sheetens koordinater och inte den utvalda spritens koordinater.
Det här kan också användas genom att t.ex. se vilken färg som finns under en racerbil i ett racerspel. Anta att bakgrunden är en separat sprite som heter "bakgrunden" som fyllts med en enda bild på en gräsplan med en racerbana på som är svart. Man har en röd och en blå bil. Kör man på det gröna utanför racerbanan har man ett värde av grönt under bilen som är 65-90. Då kan man anta att hastigheten är halverad. Koden blir då:
//Räkna ut färg under blå bil sf::Color colorB = bakgrunden.GetPixel (bluecar.sprite.GetPosition().x,bluecar.sprite.GetPosition().y); red = (int)colorB.r; green = (int)colorB.g; blue = (int)colorB.b; vinkel2 = bluecar.sprite.GetRotation();//Vilken vinkel står bilen i? double bilhastighet1 = bluecar.hastighet * 0.002; if (green > 65 && green < 90) //står på gräset bilhastighet1 = (bluecar.hastighet * 0.002) /2; else bilhastighet1 = bluecar.hastighet * 0.002; newxB= sin(vinkel2 * M_PI/180.0) * bilhastighet1; newyB= cos(vinkel2 * M_PI/180.0) * bilhastighet1; bluecar.sprite.Move(newxB, newyB); //åk iväg
Sedan är det bara att köra iväg med bilen. Eftersom färgen är en struct, måste man skapa en ny instans för varje fordon. Till den röda bilen blir det t.ex.:
const sf::Color colorR = bakgrunden.GetPixel (redcar.sprite.GetPosition().x,redcar.sprite.GetPosition().y); red = (int)colorR.r; green = (int)colorR.g; blue = (int)colorR.b;
I exemplet använder jag samma red, green och blue för bägge bilarna efter det att jag skapat dem som vatiabler i Main. Det är dock smartare att lägga in re, green och blue inuti en klass istället och mata in värdena till dem i klassen allteftersom man kör runt med de olika bilarna på spelplanen.
Dold bakgrund
redigeraAtt hitta färgen under en speciell punkt är också ett väldigt enkelt sätt att t.ex. kontrollera om man krockat i ett plattformsspel. I exemplet här ovan kallade vi bakgrunden för "bakgrunden" rätt och slätt och det är den spritens bild vi använder osss av i kodexemplet:
sf::Color colorB = bakgrunden.GetPixel (bluecar.sprite.GetPosition().x,bluecar.sprite.GetPosition().y);
men anta att vi har en annan sprite som heter "dold_bakgrund" och som vi aldrig ritar ut. På den kan vi ha alla saker man kan hoppa på i plattformsspelet utritade med tydliga färger. Anta att alla plattformarna är knallröda och stationära, då gör du istället
sf::Color colorB = dold_bakgrunden.GetPixel (bluecar.sprite.GetPosition().x,bluecar.sprite.GetPosition().y); //bortklippt kod if (red == 255) {collision = true;} else {collision = false;}
Dvs. om den färg man testar mot är knallröd så har vi krockat med en plattform som är knallröd. Det gör detsamma om plattformarna inte är knallröda på den bild som visas utåt mot spelaren.
Skall man ta detta ett steg längre mäter man på olika punkter. I ett plattformsspel kan man kolla mitten på över- och underkanten samt mitten på höger- och vänster sida. Om någon av de punkterna ligger på en röd yta har man krockat.
Detta system fungerar endast på saker som är fasta i spelet. Saker som tillverkas nya, som kanonkulor t.ex. fungerar det inte på, om man inte skulle få för sig att göra även dem knallröda.
Om spelpjäserna är små är det lättast att bara mäta mot mittenpunkten. Det kan ge litet udda bieffekter men fungerar vanligtvis tillräckligt bra.
Komplett kod med både Cirkular och Box collision detect
redigera#include <iostream> #include <vector> //Lägg till när du använder en vector #include <SFML\System.hpp> #include <SFML\Graphics.hpp> #include <SFML\Window.hpp> #include <SFML\Audio.hpp> #define M_PI 3.14159265358979323846 /* pi som statisk konstant*/ #define SFML_STATIC //Se till så att det inte behövs extra DLL-filer using namespace std; //Class för skottet class Pansarvagnsskott { public : Pansarvagnsskott(double ut_hastighet, double ut_vinkel, bool ut_kan_ses); //Konstruktion ~Pansarvagnsskott(){}; double target_x; double target_y; double hastighet; double vinkel; bool kan_ses; sf::Sprite Sprite; // en per instans sf::Image Image; //bildfilshållaren }; Pansarvagnsskott::Pansarvagnsskott(double ut_hastighet, double ut_vinkel, bool ut_kan_ses) { hastighet=ut_hastighet; vinkel=ut_vinkel; kan_ses = ut_kan_ses; Sprite.SetRotation(vinkel);//Ge den rätt vinkel redan från start Sprite.SetImage(Image); //ge bilden till spriten //Obs, ingen image här - det ger vita rutor i en vector } class Explosion { public: Explosion(bool ut_playing, bool ut_has_played, double ut_starttid); //Konstruktion ~Explosion(){};//Destruktor sf::Sprite Sprite; // en per instans sf::Image Image; //bildfilshållaren //var skall explosionen ske? double target_x; double target_y; //Var i sprite sheeten skall vi börja kopiera bilder int topkoordinatX; int topkoordinatY; double starttid; //Vid vilket ögonblick skall explosionen starta? bool playing; //visas den upp? bool has_played; //Har den visats färdigt? void get_explosionsbild() //Ta ut rätt koordinater i sprite sheeten för visning av explosion { //Ta fram rätt bild Sprite.SetSubRect(sf::IntRect(topkoordinatX,topkoordinatY, topkoordinatX + 64,topkoordinatY + 64)); //Funktionen ger övre vänstra hörnets koordinater //då kan vi själva räkna ut nedre högra hörnet eftersom bilderna är 64 x 64 stora } }; Explosion::Explosion(bool ut_playing, bool ut_has_played, double ut_starttid) //Initiering { playing = ut_playing; //Avgör om animationen spelar. Ange false som standard has_played = ut_has_played; //Ställs från funktion utanför klass, false från början starttid = ut_starttid; //När börjar explosionen Sprite.SetImage(Image); //ge bilden till spriten //Obs, ingen image här - det ger vita rutaor i en vector } //En moderklass för alla fordon i spelet class fordon { public: //Konstruktordeklaration, definition utanför klassdeklarationen fordon(double hastighet, double spelare_x, double spelare_y, int bensin, int pansar);//startvärden //Destruktion ~fordon(){}; double hastighet; //Hur snabb är den double spelare_x; // var är den i sidled i programmet double spelare_y; //var är den i höjdled i programmet int bensin; //Hur långt kan den köra int pansar; //Hur mycket tål den sf::Sprite sprite; //Används i grafiskt läge }; //Konstruktionsdeklaration, fordon-------------------------------------------------- fordon::fordon (double ut_hastighet, double ut_spelare_x, double ut_spelare_y, int ut_bensin, int ut_pansar) { hastighet=ut_hastighet; spelare_x=ut_spelare_x; spelare_y=ut_spelare_y; bensin = ut_bensin; pansar = ut_pansar; //Också vill vi se att det verkligen skapas ett fordon när underklasserna skapas std::cout << "Ett fordon rullar ut från fabriken!" << std::endl; } //Skapa en klass som ärver fordon class stridsvagn : public fordon { public: int ammunition; // Avgör hur många skott som finns i stridsvagnen. stridsvagn(double ut_hastighet, double ut_spelare_x, double ut_spelare_y, int ut_bensin, int ut_pansar): fordon(ut_hastighet, ut_spelare_x, ut_spelare_y, ut_bensin, ut_pansar) { ammunition=100; } //Destruktion ~stridsvagn(){}; //Funktioner void skott() { std::cout << "Skott kommer!" << std::endl; ammunition = ammunition-1; std::cout << "Skott kvar=" <<ammunition << std::endl; } }; void explosionsbildruta(double speltid, Explosion &Klassinstans); //Funktion som tar fram x och y koordinat på sprite sheet för explosionsbilder //och som stänger av animationen om fler än 25 bilder visas bool check_hit(double x, double y); //Funktion som kollar om kulan träffat ett mål //Kolla om kulan har träffat en annan spelare*********************************************************** bool Cirkular_check_hit(sf::Sprite &kulsprite, sf::Sprite &maolsprite); //se om kulan träffat //Kolla om box passerat box mellan två spelare bool Box_check_hit(sf::Sprite &kulsprite, sf::Sprite &maolsprite); double mot_koordinat(sf::Sprite &sprite, sf::RenderWindow &window, double position_X, double position_Y, int gradjusterare); //Funktion som roterar pansarvagnen mot en speciell koordinat //******************************************************PROGRAMSTART******************************************************** int main (int argc, char *argv) { //main startar //För att kunna styra animationer i spelet måste vi kunna hålla koll på tiden. //Det finns många sätt att göra det på och det här är det sämsta //men det fungerar och är lätt att förstå. sf::Clock spelklocka; //Skapa en spelklocka spelklocka.Reset(); //Ställ den på 0 //Övriga variabler double explosionsstarttid = 0.0f; //bool visaexplosion = false; //skall den visas? //Variabler för att hålla ordning i listorna int skott_i=0; // avgör var vi är i kanonkulelistan int si = 0; //räknare int exp_i=0; //0-100 avgör var vi är i explosionslistan int ei = 0; //räknare //Vector för pansarvagnsskott std::vector<Pansarvagnsskott> vSkottlista; //Vector för explosioner std::vector<Explosion> vExplosionslista; //Bilderna för skotten, dels en granat och dels en explosion //Ligger laddningen inom klassen skapas vita rutor istället sf::Image KanonkulaImage; //kanonkula if (!KanonkulaImage.LoadFromFile("Eight-Ball-icon.png")) //Programmet försöker att ladda in en bildfil i det tomma bildobjektet std::cout << "Kan inte hitta bilden: Eight-Ball-icon.png " << std::endl; sf::Image ExplosionImage; //explosion if (!ExplosionImage.LoadFromFile("xplosion17.png")) //Programmet försöker att ladda in en bildfil i det tomma bildobjektet std::cout << "Kan inte hitta bilden: xplosion17.png " << std::endl; //Olika värden för att styra stridsvagnen float newx = 0.0f; //ditt stridsvagnen skall gå float newy = 0.0f; //dit stridsvagnen skall gå float speed = 2.0f; //stridsvagnens hastighet. double vinkel =0.0f; //Stridsvagnens vinkel //Skapa först en stridsvagn //stridsvagn(int ut_hastighet, double ut_spelare_x, double ut_spelare_y, int ut_bensin, int ut_pansar) stridsvagn stridsvagn1(0.01f, 100.0,100.0,100,100); //Skapa sedan kanonen stridsvagn kanon1(0.0f, 700.0, 500.0, 0, 50); //Skapa bild sf::Image bild; //skapa en tom bildhållare som heter bild if (!bild.LoadFromFile("MH_Tank.png")) return EXIT_FAILURE; //fyll den tomma bildhållaren med bilden MH_tank.png stridsvagn1.sprite.SetImage(bild); //Skapa bild för kanonen sf::Image kanonbild; //skapa en tom bildhållare som heter bild if (!kanonbild.LoadFromFile("AT_Type_1_Cannon.png")) return EXIT_FAILURE; //fyll den tomma bildhållaren med kanonbilden kanon1.sprite.SetImage(kanonbild); //nu har vi en stridsvagn, Placera ut stridsvagnen på spelplanen stridsvagn1.sprite.SetPosition(stridsvagn1.spelare_x,stridsvagn1.spelare_y); //Placera också ut kanonen på spelplanen kanon1.sprite.SetPosition(kanon1.spelare_x,kanon1.spelare_y); //Det är jobbigt att inte riktigt veta var mitten är på stridsvagnen //Ändra positionen där stridsvagnen är från övre vänstra hörnet till mitt på bilden. stridsvagn1.sprite.SetCenter(stridsvagn1.sprite.GetSize().x / 2 , stridsvagn1.sprite.GetSize().y / 2); //Gör samma med kanonen kanon1.sprite.SetCenter(kanon1.sprite.GetSize().x / 2 , kanon1.sprite.GetSize().y / 2); //Ladda in de två ljuden //Ljud för att avlossa kanonen sf::SoundBuffer kanonljud; //skapa en ljudbuffer/hållare if (!kanonljud.LoadFromFile("explosion-03.wav")) //ladda in en fil i hållaren { std::cout << "Kan inte hitta ljudfilen: explosion-03.wav " << endl; } sf::Sound kanoneffekt; //skapa ett ljud i spelet för kanonen som vi döper till kanoneffekt kanoneffekt.SetBuffer(kanonljud); // Ladda in ljudfilens värden i ljudet så att det går att spela upp. //Ljud när granaten exploderar sf::SoundBuffer explosionsljud; //skapa en ljudbuffer/hållare if (!explosionsljud.LoadFromFile("bomb-03.wav")) //ladda in en fil i hållaren { std::cout << "Kan inte hitta ljudfilen: bomb-03.wav " << endl; } sf::Sound ljudeffekt; //skapa ett ljud i spelet som vi döper till ljudeffekt ljudeffekt.SetBuffer(explosionsljud); // Ladda in ljudfilens värden i ljudet så att det går att spela upp. /************** Programstart ***********************************************************/ sf::RenderWindow App(sf::VideoMode(800, 600, 32), "Test av pansarvagn"); // Skapa fönstret vi skall testa explosionerna i while (App.IsOpened()) { //while 1 startar, spelet körs sf::Event Event; //kolla om mus/tangentbord används while (App.GetEvent(Event)) { //while 2 börjar, kontrollerar händelser if (Event.Type == sf::Event::MouseButtonPressed) // En musknapp har tryckts ner {//musknapp, vilken som helst if (Event.MouseButton.Button == sf::Mouse::Left) { //vänster musknapp //Vrid stridsvagnen i rätt vinkel mot_koordinat(stridsvagn1.sprite, App, App.GetInput().GetMouseX(), App.GetInput().GetMouseY(), 90); stridsvagn1.spelare_x = App.GetInput().GetMouseX(); //Ge stridsvagnen en X-koordinat att gå mot stridsvagn1.spelare_y = App.GetInput().GetMouseY(); //Ge stridsvagnen en Y-koordinat att gå mot } //vänster musknapp }//slut mus, vilken knapp som helst //Rotera kanonen med höger/vänster piltangent if (App.GetInput().IsKeyDown(sf::Key::Right)) kanon1.sprite.Rotate(- 10); //medsols if (App.GetInput().IsKeyDown(sf::Key::Left)) kanon1.sprite.Rotate(+ 10); //motsols if (Event.Type == sf::Event::Closed) //kryssat på [x] symbolen? stäng programmet App.Close(); if ((Event.Type == sf::Event::KeyPressed) && (Event.Key.Code == sf::Key::Escape)) { //Avsluta programmet //Radera vectorer för att ge tillbaka minne vExplosionslista.clear(); vSkottlista.clear(); App.Close();//avsluta programmet om man klickar på ESC } //avsluta programmet /************************* Kanonskott *******************************************************************************************/ // Här kommer koden för att avlossa kanonen! // Ett explosionsljud spelas upp // Mängden ammunition i stridsvagnen räknas ner ett steg // En ny kanonkula skapas och läggs in i en array // Kanonkulans mittpunkt sätts mitt på kulan istället för i övre högra hörnet // Kanonkulan har sin mittpunkt på samma punkt som pansarvagnen har sin // För varje kula skapar vi en potentiell explosion med samma indexnummer // Explosionens mittpunkt placeras på samma ställe som kulans mittpunkt // Räknaren skott_i flyttar ett steg upp och väntar på nästa kula // Slutligen synkroniseras skottlistan med explosionslistan /********************************************************************************************************************/ if ((Event.Type == sf::Event::KeyPressed) && (Event.Key.Code == sf::Key::Space)) {//Spacetangent nertryckt kanoneffekt.Play(); //spela upp kanonljudet stridsvagn1.skott(); //Skjut med stridsvagnens funktion //Nytt- vi använder vektorer vSkottlista.push_back( Pansarvagnsskott(1.0f,stridsvagn1.sprite.GetRotation(), true) ); //Lägger en kula i vector //Skapa ett nytt skott med: //hastighet 1.0f och vinkel exakt samma som stridsvagnen och true= den syns. //vSkottlista.at(skott_i).Sprite.SetCenter(vSkottlista.at(skott_i).Sprite.GetSize().x / 2 , vSkottlista.at(skott_i).Sprite.GetSize().y / 2); //Sätt koordinaten mitt på vSkottlista.at(skott_i).Sprite.SetCenter(8,8); //Sätt koordinaten mitt på, bilden är 16x16 stor //Kan även skrivas vSkottlista[skott_i].Sprite.SetCenter(vSkottlista.at(skott_i).Sprite.GetSize().x / 2 , vSkottlista.at(skott_i).Sprite.GetSize().y / 2); //Sätt koordinaten mitt på vSkottlista.at(skott_i).Sprite.SetPosition(stridsvagn1.sprite.GetPosition().x,stridsvagn1.sprite.GetPosition().y); //Placera ut den på samma punkt som pansarvagnen vExplosionslista.push_back( Explosion(false, false, spelklocka.GetElapsedTime()) );//lägger en explosion i vector vExplosionslista.at(skott_i).Sprite.SetCenter(32,32); //Hela bilden är 64 x 64 //Slutligen, ge bilderna utifrån vExplosionslista.at(skott_i).Sprite.SetImage(ExplosionImage); //ge bilden till spriten vSkottlista.at(skott_i).Sprite.SetImage(KanonkulaImage); //ge bilden till spriten //Både nytt och gammalt skott_i++; //Räkna upp till nästa plats i kön för nästa skott exp_i = skott_i; //Synkronisera de två listorna (skott och explosioner) så att de alltid pekar på samma postnummer i //båda listorna. Det behövs sedan för att rita ut korrekt animation på rätt plats }//Spacetangent nertryckt } //while 2 slutar //Slutligen visar vi upp ändringarna om och om igen många gånger i sekunden App.Clear(sf::Color(0, 100, 0)); //rensa allt i fönstret och ersätt med grönfärg /**************************** Flytta på skotten på spelplanen ****************************************/ if ( vSkottlista.empty() == false ) { // om vectorn inte är tom for (si=0; si < vSkottlista.size(); si++) //for istället för while eftersom vi inte vet antalet poster längre { //Gå igenom hela listan vinkel = vSkottlista.at(si).vinkel; // 0 = rakt upp, startposition. Motsols 0-360 grader. newx= sin(vinkel*M_PI/180.0) * vSkottlista.at(si).hastighet; //Flytt i x-led newy= cos(vinkel*M_PI/180.0) * vSkottlista.at(si).hastighet; //Flytt i Y-led vSkottlista.at(si).Sprite.Move(newx*-1, newy*-1); //gånger * -1 eftersom rakt upp är 0 och inte 90 grader /****************************************************************************************************/ // Olika rutiner för att se om mman fått en träff: // En standardrutin som kollar om kulan är 50 pixlar från kanten, bra att ha för enkel felkontroll // En för att se om radiena går in i varandra, bra för runda föremål // En för att se om två boxar går in i varandra, bra för långsmala föremål // En för att avgöra om en träff skett beroende på färg. //**************************************************************************************************/ //Kolla träff om kulan är 50 pixlar från kanten // if (check_hit(vSkottlista.at(si).Sprite.GetPosition().x, vSkottlista.at(si).Sprite.GetPosition().y) == true) //kolla träff för cirkular hit mellan kulan och kanonen if ( Cirkular_check_hit(vSkottlista.at(si).Sprite, kanon1.sprite)== true) //se om kulan träffat ///if (Box_check_hit(vSkottlista.at(si).Sprite, kanon1.sprite) == true)//se om kulan träffat {//träff //Enklast möjligt; samma explosionsljud används hela tiden if (vSkottlista.at(si).hastighet > 0) {//kulan är i rörelse vExplosionslista.at(si).playing = true; //börja spela upp animationen. vExplosionslista.at(si).target_x = vSkottlista.at(si).Sprite.GetPosition().x; //x och y vExplosionslista.at(si).target_y = vSkottlista.at(si).Sprite.GetPosition().y; if(ljudeffekt.GetStatus() != sf::Sound::Status::Playing) { ljudeffekt.Play(); //spela upp ljudet om den inte spelas redan // while(ljudeffekt.GetStatus() == sf::Sound::Status::Playing) {} } else { ljudeffekt.Stop(); //stoppa och påbörja ljudet igen ljudeffekt.Play(); //while(ljudeffekt.GetStatus() == sf::Sound::Status::Playing) {} } }//kulan är i rörelse vSkottlista.at(si).kan_ses=false; //Gör så att kulan försvinner vSkottlista.at(si).hastighet = 0; //ge kulan hastighet 0 }//träff if (vSkottlista.at(si).kan_ses==true) App.Draw(vSkottlista.at(si).Sprite); }//gå igenom hela listan }//om vectorn inte är tom //Var skall pansarvagnen gå? vinkel = stridsvagn1.sprite.GetRotation(); // 0 = rakt upp, startposition. Motsols 0-360 grader. newx= sin(vinkel*M_PI/180.0) * stridsvagn1.hastighet; newy= cos(vinkel*M_PI/180.0) * stridsvagn1.hastighet; stridsvagn1.sprite.Move(newx*-1, newy*-1); //Multiplikation * -1 eftersom rakt upp är 0 och inte 90 grader App.Draw(stridsvagn1.sprite); //Rita upp stridsvagnen App.Draw(kanon1.sprite); //Rita upp kanonen //********** Visa upp animationerna*******************************************************************************************// // Dags att rita upp animationerna av explosionerna // Vi gör det sist av allt så att explosionerna ligger ovanpå // både kanonkulor och stridsvagnar. // // Precis som vi går igenom listan av kulor som skall flyttas // går vi igenom listan av explosioner för att se om värdet "playing" är true/sant // och om den inte spelats upp klart. Om båda är sant visas bildrutan upp // // Funktionen "explosionsbildruta" används för att välja ut rätt x/y koordinater // för rätt bild i sprite sheeten. //__________________________VECTORSÄTTET___________________________----- if ( vExplosionslista.empty() == false ) { // om vectorn inte är tom for (ei=0; ei < vExplosionslista.size(); ei++) //for istället för while eftersom vi inte vet antalet poster längre { //Gå igenom hela listan if (vExplosionslista.at(ei).playing == true && vExplosionslista.at(ei).has_played == false ) //man är under 26 bilder { explosionsbildruta(spelklocka.GetElapsedTime(), vExplosionslista.at(ei)); //Ge spritens bild rätt x och y koordinater, funktion vExplosionslista.at(ei).get_explosionsbild(); //Välj rätt bild till explosionen, funktion inom klassen Explosion vExplosionslista.at(ei).Sprite.SetPosition(vExplosionslista.at(ei).target_x, vExplosionslista.at(ei).target_y); //Placera ut den på rätt ställe App.Draw(vExplosionslista.at(ei).Sprite); //Rita ut bilden på explosionen } } //Gå igenom hela listan }//Om vectorn inte är tom //**********Slut på att visa upp animationerna***************************************************** App.Display(); //visa upp ändringarna för användaren //Radera slutligen alla kulor och explosioner som inte används if ( vExplosionslista.empty() == false ) { // om vectorn inte är tom for (ei=0; ei < vExplosionslista.size(); ei++) //for istället för while eftersom vi inte vet antalet poster längre { //Gå igenom hela listan if (vExplosionslista.at(ei).playing == false && vExplosionslista.at(ei).has_played == true) //man är över 26 bilder { //animationen är klar vExplosionslista.erase(vExplosionslista.begin() + ei); //Raderar ut den färdiga explosionen vSkottlista.erase(vSkottlista.begin() + ei); //Radera motsvarane kanonkula skott_i = skott_i - 1; //Återställ antalet poster exp_i = skott_i; //I bägge variablerna } //animationen klar } //Gå igenom hela listan } // om vectorn inte är tom } //while 1 slutar return 0; } //main slutar********************************************************************************** //Funktioner som används i spelet void explosionsbildruta(double speltid, Explosion &Klassinstans) { //De koordinater vi är ute efter int utx = 0; int uty = 0; double tidsomgaott = speltid-Klassinstans.starttid; //Hur lång tid har det gått sedan explosionen startade //Starttiden finns lagrad inuti klassen double visningstid = 0.05; //0.05 sekund mellan varje bild //25 bilder = värden mellan 0.25-1.25 vid 0.05 //Räkna ut Y koordinaten if (tidsomgaott <= (visningstid * 5)) //rad 1 uty = 0; else if(tidsomgaott > visningstid * 5 && tidsomgaott <= visningstid *10)//rad 2 uty = 64; else if(tidsomgaott > visningstid * 10 && tidsomgaott <= visningstid *15)//rad 3 uty = 128; else if(tidsomgaott > visningstid * 15 && tidsomgaott <= visningstid *20)//rad 4 uty = 192; else if(tidsomgaott > visningstid * 20 && tidsomgaott <= visningstid *25)//rad 5 uty = 256; //Räkna ut x koordinaten //Rad 1 if (tidsomgaott >= 0.0 && tidsomgaott < visningstid) utx = 0; //ruta 1 else if (tidsomgaott > visningstid && tidsomgaott < (visningstid * 2)) utx= 64; else if (tidsomgaott > (visningstid * 2) && tidsomgaott < (visningstid * 3)) utx= 128; else if (tidsomgaott > (visningstid * 3) && tidsomgaott < (visningstid * 4)) utx= 192; else if (tidsomgaott > (visningstid * 4) && tidsomgaott < (visningstid * 5)) utx= 256; //Rad 2 else if (tidsomgaott > visningstid * 5 && tidsomgaott < visningstid *6) utx= 0; //ruta 1 else if (tidsomgaott > visningstid * 6 && tidsomgaott < (visningstid * 7)) utx= 64; else if (tidsomgaott > (visningstid * 7) && tidsomgaott < (visningstid * 8)) utx= 128; else if (tidsomgaott > (visningstid * 8) && tidsomgaott < (visningstid * 9)) utx= 192; else if (tidsomgaott > (visningstid * 9) && tidsomgaott < (visningstid * 10)) utx= 256; //Rad 3 else if (tidsomgaott > visningstid * 10 && tidsomgaott < visningstid *11) utx= 0; //ruta 1 else if (tidsomgaott > visningstid * 11 && tidsomgaott < (visningstid * 12)) utx= 64; else if (tidsomgaott > (visningstid * 12) && tidsomgaott < (visningstid * 13)) utx= 128; else if (tidsomgaott > (visningstid * 13) && tidsomgaott < (visningstid * 14)) utx= 192; else if (tidsomgaott > (visningstid * 14) && tidsomgaott < (visningstid * 15)) utx= 256; //Rad4 else if (tidsomgaott > visningstid * 15 && tidsomgaott < visningstid * 16) utx= 0; //ruta 1 else if (tidsomgaott > visningstid * 16 && tidsomgaott < (visningstid * 17)) utx= 64; else if (tidsomgaott > (visningstid * 17) && tidsomgaott < (visningstid * 18)) utx= 128; else if (tidsomgaott > (visningstid * 18) && tidsomgaott < (visningstid * 19)) utx= 192; else if (tidsomgaott > (visningstid * 19) && tidsomgaott < (visningstid * 20)) utx= 256; //Rad5 else if (tidsomgaott > visningstid * 20 && tidsomgaott < visningstid *21) utx= 0; //ruta 1 else if (tidsomgaott > visningstid * 21 && tidsomgaott < visningstid * 22) utx= 64; else if (tidsomgaott > visningstid * 22 && tidsomgaott < visningstid * 23) utx= 128; else if (tidsomgaott > visningstid * 23 && tidsomgaott < visningstid * 24) utx= 192; else if (tidsomgaott > visningstid * 24 && tidsomgaott <= visningstid * 25) utx= 256; else if (tidsomgaott > (visningstid * 25)) Klassinstans.has_played = true; //Visa bara 25 bilder, sluta spela upp animationen Klassinstans.topkoordinatX = utx; //skriv in koordinaterna direkt i klassen Klassinstans.topkoordinatY = uty; //låt klassen beräkna bilden } //funktionsslut //Kolla om kulan har träffat en vägg*********************************************************** bool check_hit(double x, double y) //se om kulan träffat { bool kulan_har_traeffat = false; if (x < 50 || x > 750) kulan_har_traeffat = true; if (y < 50 || y > 550) kulan_har_traeffat = true; return kulan_har_traeffat; } //Kolla om kulan har träffat en annan spelare*********************************************************** bool Cirkular_check_hit(sf::Sprite &kulsprite, sf::Sprite &maolsprite) //se om kulan träffat { bool kulan_har_traeffat = false; //returvärde double avstaond = 0.0; //Avståndet mellan två sprites mittpunkter double traeffavstaond = 0.0; //sammanlagda summan av två sprites radie double tempx; //sträckan i x-led double tempy; //sträckan i y-led //Ta fram koordinaterna för de två spritesen. //kulax och kulay är kanonkulans koordinater //maolx och maoly är måltavlans koordinater double kulax = kulsprite.GetPosition().x; double kulay = kulsprite.GetPosition().y; double maolx = maolsprite.GetPosition().x; double maoly = maolsprite.GetPosition().y; //Ta fram kulansradie, 16 x 16 pixels stor double kula_radie = (1.4142135623730950488016887242097 * 16) / 2; //Ta fram målets radie, 64 x 64 pixels stor double maol_radie = (1.4142135623730950488016887242097 * 32) / 2; //Så när träffar dem? vilket avstånd minst från varandra? traeffavstaond = kula_radie + maol_radie; if (kulax == maolx && kulay != maoly ) //de står på samma linje i x-led { avstaond = kulay - maoly; if (avstaond <= traeffavstaond) //inom träffavstånd kulan_har_traeffat = true; } else if (kulay == maoly && kulax != maolx ) //de står på samma linje i y-led { avstaond = kulax - maolx; if (avstaond <= traeffavstaond) //inom träffavstånd kulan_har_traeffat = true; } else if(kulax != maolx && kulay != maoly) //De står på olika platser både x och y { //Räkna med pythagoras tempx = kulax-maolx; tempy = kulay-maoly; // Räkna ut hypotenusan/avståndet genom att ta // kvadratroten ur ena sträckan ^2 + andra sträckan ^2 avstaond = sqrt ( tempx * tempx + tempy * tempy); if (avstaond <= traeffavstaond) //inom träffavstånd kulan_har_traeffat = true; } //Räkna med pythagoras else if (kulax == maolx && kulay == maoly) //De står på exakt samma plats kulan_har_traeffat = true; return kulan_har_traeffat; } //Kolla om kulan har träffat kanten av en annan spelare*********************************************************** bool Box_check_hit(sf::Sprite &kulsprite, sf::Sprite &maolsprite) //se om kulan träffat { bool kulan_har_traeffat = false; //returvärde //Ta fram koordinaterna för de två spritesen. //kulax och kulay är kanonkulans koordinater //maolx och maoly är måltavlans koordinater double kulax = kulsprite.GetPosition().x; double kulay = kulsprite.GetPosition().y; double maolx = maolsprite.GetPosition().x; double maoly = maolsprite.GetPosition().y; //Kulan är 16 x 16 och målet är 64 x 64 // innanför vänster kant innanför höger kant innanför över kant innanför neder kant if ((kulax + 8 > (maolx - 32)) && (kulax - 8 < maolx + 32) && kulay + 8 > maoly -32 && kulay - 8 < maoly + 32 ) kulan_har_traeffat = true; //returvärde return kulan_har_traeffat; } //Rotera stridsvagn mot en speciell koordinat ***************************************************************************************** double mot_koordinat(sf::Sprite &sprite, sf::RenderWindow &window, double position_X, double position_Y, int gradjusterare) { double spritensvinkel; //Den vinkel vår sprite stannar på double PI = 3.14159265358979323846; //Utifall att pi inte är definierad if (position_X <= sprite.GetPosition().x) //Kollar om inte x eller y är likadana sprite.SetRotation (((-1* 360 / PI * (atan2(static_cast<double> (sprite.GetPosition().y - position_Y) , static_cast<double> (sprite.GetPosition().x - position_X))))/2)+gradjusterare); else sprite.SetRotation (((-1* 360 / PI *(atan2(static_cast<double> (sprite.GetPosition().y - position_Y) , static_cast<double> (sprite.GetPosition().x - position_X))))/2)+gradjusterare); window.Draw(sprite); //Rita ut ändringen spritensvinkel = sprite.GetRotation(); //ta fram vinkeln return spritensvinkel; //Skicka tillbaka vinkeln } //***************************************PROGRAMSLUT*****************************************************************//