Oppgaven går ut på å implementere spillet Slagskip (eng: Battleship) (http://en.wikipedia.org/wiki/Battleship_(game)) som skal kunne kjøres i konsollen. Det er to spillere, med hvert sitt brett med utplasserte skip. Ulike typer skip har ulik størrelse og fasong. Målet er å senke alle motstanderens skip ved å bombe rutene som skipene dekker. Motstanderens skip er skjult for spilleren i starten, men siden en etter å ha bombet en rute får vite om en har truffet eller ikke, så får en oversikt over motstanderens brett etterhvert. Et skip er senket når alle rutene det dekker er truffet. Spillet er vunnet når en spiller har senket alle motstanderens skip.
I vår implementasjon er det ikke meningen at en spiller skal kunne se hvor sine egne skip er plassert. Siden begge spillerne spiller på samme maskin, så er jo ikke det særlig gunstig.
Et Slagskip-brett er et rutenett på 10 x 10 ruter. Det finnes fem forskjellige skip, med følgende dimensjoner:
Type skip | Dimensjoner |
---|---|
Hangarskip | 2 x 3 |
Slagskip | 1 x 4 |
Ubåt | 1 x 3 |
Fregatt | 1 x 3 |
Patruljebåt | 1 x 2 |
Merk at skip kan orienteres i to retninger, altså ligge enten horisontalt eller vertikalt på brettet.
Krav
Programmet må bestå av minst tre klasser (de trenger ikke ha samme navn som under):
Cell – en klasse for å representere innholdet i og tilstanden til en rute på brettet.
BattleshipBoard – en klasse for selve spillbrettet.
BattleshipProgram – et hovedprogram for interaksjon med spilleren.
Brett-klassen må kunne initialiseres med en String på standard-formatet, som beskrevet under.
Spillet skal støtte to spillere med hvert sitt brett.
Spillet må veksle mellom de to spillerne og skrive ut til konsollet hvilken spiller sin tur det er og brettet med oversikt over bom og treff for denne spilleren. Så skal de kunne skrive inn koordinatene for neste bombe.
En spiller skal ikke få lov til å bombe en rute han allerede har bombet.
Etter at en bombe er sluppet så skal det komme en tekstlig beskjed i konsollen om det var treff eller ikke, i tillegg til å kunne se det på motstanderens brett.
Når en av spillerene har vunnet skal spillet stoppes, og det skal skrives til konsoll hvem som vant.
Eksempelbrett
I tabellen under kan du finne tre eksempelbrett. BattleshipBoard-klassen må ha en konstruktør som tar inn en String, og konstruerer et brett av denne strengen. Eksempelstrengene består av punktum og X, der punktum betyr hav og X betyr skip. "\n" betyr linjeskift. Du må gjerne støtte egne tegn for hver type skip, bare du bruker bokstaver (A-Z). Hvis du lagrer brett for spill underveis, så bruk små bokstaver for ruter som er truffet og store for de som ikke er. Du trenger ikke bruke disse eksempelbrettene, du må gjerne lage dine egne.
Eksempelbrett | Ser slik ut |
---|---|
SAMPLE_LEVEL1 | "...XX........XX..XXX...XX..........................X.........X...XXX...X.........X..XX.............." |
SAMPLE_LEVEL2 | ".XXX.......XXX....X.........X.........X....XX...X.............X..XXX....X.........X................." |
SAMPLE_LEVEL3 | "................................XX.........XXX........XXXX.....XXX.....XXX.......XXX................" |
Tips: Lag en egen klasse Levels.java som har disse eksempelbrettene som felt (tilstand). De kan være public static final. final betyr at verdien til variabelen ikke kan endres etter den er initialisert. Se /sokoban/SampleLevels.java for inspirasjon.
Forslag til grafisk implementasjon og gjennomførelse av to runder:
Tegn | Forklaring |
---|---|
~ | Hav. Uskutt rute. Ligner på bølge, ikke sant? |
. | Skutt rute, men ingen skip truffet. |
X | Skip truffet. |
Input fra brukeren er markert i grønt.
Player 1's turn: | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | 2 2 That's a miss... Player Two's turn: | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | 5 6 That's a hit! Player One's turn: | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ . ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | 2 5 That's a miss... Player Two's turn: | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ X ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ | | ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ |
JavaFX-app
Konsollbasert IO er greit for å teste spillet, men det er kanskje artigere å lage en ordentlig app, med grafisk visning av rutenettet og bruk av musa for å klikke på ruter du vil skyte. Vi har laget en JavaFX-basert app, som du kan laste ned og koble til din Battleship-klasse. App-koden forutsetter at spill-logikken er implementert i en klasse Battleship som implementerer IBattleship-grensesnittet (i pakken games.battleship). Med bruk av grensesnitt, så gjøres det klart hvilke metoder appen trenger og som du må implementere.
Last først ned IBattleship-grensesnittet fra /games/battleship/IBattleship.java, og programmer din spill-klasse slik at den implementerer dette grensesnittet. Så laster du ned app-klassene (og filene), så du kan prøve om det virker. Husk at du må endre en linje i BattleshipFX.fxml slik at den peker på klassen din som implementerer IBattleship-grensesnittet.
Fil-tre som må lastes ned evt. importeres med Digital Compendium-panelet
emfs /games/ #java #package FxmlApp.java @ git@github.com:hallvard/javafx/games/src/games/FxmlApp.java; FxmlBuilderFactory.java @ git@github.com:hallvard/javafx/games/src/games/FxmlBuilderFactory.java; FxmlGame.java @ git@github.com:hallvard/javafx/games/src/games/FxmlGame.java; imagegrid/ #java #package ImageGrid.java @ git@github.com:hallvard/javafx/games/src/games/imagegrid/ImageGrid.java; ImageGridGame.java @ git@github.com:hallvard/javafx/games/src/games/imagegrid/ImageGridGame.java; ; battleship/ #java # IBattleship.java @ git@github.com:hallvard/javafx/games/src/games/battleship/IBattleship.java BattleshipFX.java @ git@github.com:hallvard/javafx/games/src/games/battleship/BattleshipFX.java BattleshipFX.fxml @ git@github.com:hallvard/javafx/games/src/games/battleship/BattleshipFX.fxml ; battleship/res hit64x64.png @ git@github.com:hallvard/javafx/games/src/games/battleship/res/hit64x64.png miss64x64.png @ git@github.com:hallvard/javafx/games/src/games/battleship/res/miss64x64.png ocean64x64.png @ git@github.com:hallvard/javafx/games/src/games/battleship/res/ocean64x64.png
Merk at du godt kan la være bruke å IBattleship-grensesnittet, men da må du omprogrammere appen, så den bruker andre metoder enn vi har forutsatt. Det er nok mer arbeid, men kanskje enda mer lærerikt?