Compare commits
	
		
			1 Commits
		
	
	
		
			c7bd7d18b7
			...
			revert-71a
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 9abf7350e4 | 
							
								
								
									
										26
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,26 +0,0 @@ | |||||||
| # Gradle |  | ||||||
| .gradle |  | ||||||
| build |  | ||||||
|  |  | ||||||
| # VSC |  | ||||||
| bin |  | ||||||
|  |  | ||||||
| # IntelliJ |  | ||||||
| *.iml |  | ||||||
| .idea |  | ||||||
| out |  | ||||||
|  |  | ||||||
| # Eclipse |  | ||||||
| .classpath |  | ||||||
| .project |  | ||||||
|  |  | ||||||
| # Libraries |  | ||||||
| *.so |  | ||||||
| *.dylib |  | ||||||
| *.dll |  | ||||||
| *.jar |  | ||||||
| *.class |  | ||||||
|  |  | ||||||
| .DS_Store |  | ||||||
| !Projekte/gradle/wrapper/gradle-wrapper.jar |  | ||||||
|  |  | ||||||
							
								
								
									
										1
									
								
								Dokumente/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1 +0,0 @@ | |||||||
| # |  | ||||||
							
								
								
									
										3
									
								
								Projekte/.gitattributes
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,3 +0,0 @@ | |||||||
| *.bat           text eol=crlf |  | ||||||
| gradlew         text eol=lf |  | ||||||
|  |  | ||||||
| @@ -1,18 +0,0 @@ | |||||||
| <component name="ProjectRunConfigurationManager"> |  | ||||||
|     <configuration default="false" name="BattleshipApp (Mac)" type="Application" factoryName="Application" |  | ||||||
|                    singleton="false"> |  | ||||||
|         <option name="MAIN_CLASS_NAME" value="pp.battleship.client.BattleshipApp"/> |  | ||||||
|         <module name="Projekte.battleship.client.main"/> |  | ||||||
|         <option name="VM_PARAMETERS" value="-XstartOnFirstThread"/> |  | ||||||
|         <option name="WORKING_DIRECTORY" value="$MODULE_WORKING_DIR$"/> |  | ||||||
|         <extension name="coverage"> |  | ||||||
|             <pattern> |  | ||||||
|                 <option name="PATTERN" value="pp.battleship.client.*"/> |  | ||||||
|                 <option name="ENABLED" value="true"/> |  | ||||||
|             </pattern> |  | ||||||
|         </extension> |  | ||||||
|         <method v="2"> |  | ||||||
|             <option name="Make" enabled="true"/> |  | ||||||
|         </method> |  | ||||||
|     </configuration> |  | ||||||
| </component> |  | ||||||
| @@ -1,18 +0,0 @@ | |||||||
| <component name="ProjectRunConfigurationManager"> |  | ||||||
|   <configuration default="false" name="BattleshipApp" type="Application" factoryName="Application" singleton="false" |  | ||||||
|                  nameIsGenerated="true"> |  | ||||||
|     <option name="MAIN_CLASS_NAME" value="pp.battleship.client.BattleshipApp"/> |  | ||||||
|     <module name="Projekte.battleship.client.main"/> |  | ||||||
|     <option name="VM_PARAMETERS" value="-Djava.util.logging.config.file=logging.properties"/> |  | ||||||
|     <option name="WORKING_DIRECTORY" value="$MODULE_WORKING_DIR$"/> |  | ||||||
|     <extension name="coverage"> |  | ||||||
|       <pattern> |  | ||||||
|         <option name="PATTERN" value="pp.battleship.client.*"/> |  | ||||||
|         <option name="ENABLED" value="true"/> |  | ||||||
|       </pattern> |  | ||||||
|     </extension> |  | ||||||
|     <method v="2"> |  | ||||||
|       <option name="Make" enabled="true"/> |  | ||||||
|     </method> |  | ||||||
|   </configuration> |  | ||||||
| </component> |  | ||||||
| @@ -1,17 +0,0 @@ | |||||||
| <component name="ProjectRunConfigurationManager"> |  | ||||||
|   <configuration default="false" name="BattleshipServer" type="Application" factoryName="Application" |  | ||||||
|                  nameIsGenerated="true"> |  | ||||||
|     <option name="MAIN_CLASS_NAME" value="pp.battleship.server.BattleshipServer"/> |  | ||||||
|     <module name="Projekte.battleship.server.main"/> |  | ||||||
|     <option name="WORKING_DIRECTORY" value="$MODULE_WORKING_DIR$"/> |  | ||||||
|     <extension name="coverage"> |  | ||||||
|       <pattern> |  | ||||||
|         <option name="PATTERN" value="pp.battleship.server.*"/> |  | ||||||
|         <option name="ENABLED" value="true"/> |  | ||||||
|       </pattern> |  | ||||||
|     </extension> |  | ||||||
|     <method v="2"> |  | ||||||
|       <option name="Make" enabled="true"/> |  | ||||||
|     </method> |  | ||||||
|   </configuration> |  | ||||||
| </component> |  | ||||||
| @@ -1,24 +0,0 @@ | |||||||
| <component name="ProjectRunConfigurationManager"> |  | ||||||
|   <configuration default="false" name="Projekte [test]" type="GradleRunConfiguration" factoryName="Gradle"> |  | ||||||
|     <ExternalSystemSettings> |  | ||||||
|       <option name="executionName" /> |  | ||||||
|       <option name="externalProjectPath" value="$PROJECT_DIR$" /> |  | ||||||
|       <option name="externalSystemIdString" value="GRADLE" /> |  | ||||||
|       <option name="scriptParameters" value="--continue" /> |  | ||||||
|       <option name="taskDescriptions"> |  | ||||||
|         <list /> |  | ||||||
|       </option> |  | ||||||
|       <option name="taskNames"> |  | ||||||
|         <list> |  | ||||||
|           <option value="test" /> |  | ||||||
|         </list> |  | ||||||
|       </option> |  | ||||||
|       <option name="vmOptions" /> |  | ||||||
|     </ExternalSystemSettings> |  | ||||||
|     <ExternalSystemDebugServerProcess>true</ExternalSystemDebugServerProcess> |  | ||||||
|     <ExternalSystemReattachDebugProcess>true</ExternalSystemReattachDebugProcess> |  | ||||||
|     <DebugAllEnabled>false</DebugAllEnabled> |  | ||||||
|     <RunAsTest>false</RunAsTest> |  | ||||||
|     <method v="2" /> |  | ||||||
|   </configuration> |  | ||||||
| </component> |  | ||||||
| @@ -1,189 +0,0 @@ | |||||||
| # Beispielprogramme des Programmierprojekts |  | ||||||
|  |  | ||||||
| Hier ist der Quellcode für das in der Einarbeitungsphase genutzte Spiel |  | ||||||
| _Battleships_ zu finden. Die Quellen bestehen aus den folgenden |  | ||||||
| Gradle-Unterprojekten: |  | ||||||
|  |  | ||||||
| * _:battleship:server_ |  | ||||||
| * _:battleship:client_ |  | ||||||
| * _:battleship:model_ |  | ||||||
| * _:battleship:converter_ |  | ||||||
| * _:common_ |  | ||||||
| * _:jme-common_ |  | ||||||
|  |  | ||||||
| _Battleships_ ist ein netzwerkbasiertes Spiel und besteht aus einem Server- und |  | ||||||
| einem Clientanteil, die in den Unterprojekten _:battleship:server_ und |  | ||||||
| _:battleship:client_ realisiert sind. Beide nutzen das Unterprojekt |  | ||||||
| _:battleship:model_, das den gemeinsamen Modellanteil enthält. |  | ||||||
|  |  | ||||||
| Die Unterprojekte _:common_ und _:jme-common_ enthalten Hilfsklassen. |  | ||||||
|  |  | ||||||
| Das Unterprojekt _:battleship:converter_ wird für _Battleships_ selbst nicht |  | ||||||
| benötigt, sondern enthält lediglich den Code, um ein im Spiel verwendetes |  | ||||||
| 3d-Modell eines Schlachtschiffs in eine _J3O_-Datei umzuwandeln, die von jME |  | ||||||
| einfacher geladen werden kann. |  | ||||||
|  |  | ||||||
| ## 1 Vorbereitung |  | ||||||
|  |  | ||||||
| Für das Programmierprojekt empfehlen wir die Verwendung von Java 20. Unter Linux |  | ||||||
| sollte [_Eclipse Temurin_](https://adoptium.net/temurin/releases/?version=20) |  | ||||||
| als JDK verwendet werden, andere JDKs können unter Linux Probleme verursachen. |  | ||||||
| Auf anderen Betriebssystemen empfehlen wir aber ebenfalls Temurin. Im Folgenden |  | ||||||
| ist beschrieben, wie Sie Temurin installieren und die Umgebungsvariable |  | ||||||
| **JAVA_HOME** richtig setzen, damit Sie Gradle (siehe unten) verwenden können. |  | ||||||
|  |  | ||||||
| ### 1.1 Installation von Temurin |  | ||||||
|  |  | ||||||
| Laden Sie [_Eclipse Temurin_](https://adoptium.net/temurin/releases/?version=20) |  | ||||||
| entsprechend Ihrem Betriebssystem und Ihrer Prozessorarchitektur herunter und |  | ||||||
| entpacken Sie das Archiv in einem Verzeichnis Ihrer Wahl auf Ihrem Rechner. |  | ||||||
|  |  | ||||||
| ### 1.2 Setzen von JAVA_HOME |  | ||||||
|  |  | ||||||
| Zur Verwendung mit Gradle muss die Umgebungsvariable **JAVA_HOME** richtig |  | ||||||
| gesetzt werden. Folgen Sie dazu den nachfolgenden Anweisungen entsprechend Ihrem |  | ||||||
| Betriebssystem: |  | ||||||
|  |  | ||||||
| * **Windows:** |  | ||||||
|  |  | ||||||
|   Öffnen Sie ihre Powershell (Core) bzw. ihr Windows Terminal mit Powershell |  | ||||||
|   (Core). Überprüfen Sie, ob die Umgebungsvariable korrekt gesetzt ist:   |  | ||||||
|   `Get-ChildItem -Path Env:JAVA_HOME`   |  | ||||||
|   Falls kein oder ein falscher Pfad gesetzt ist, setzen Sie diesen mit dem |  | ||||||
|   folgenden Kommando (in einer Zeile):   |  | ||||||
|   `[System.Environment]::SetEnvironmentVariable('JAVA_HOME','<Pfad zum SDK>',[System.EnvironmentVariableTarget]::User)` |  | ||||||
|  |  | ||||||
|   Alternativ können Sie die GUI verwenden. Unter Windows 10 klicken Sie die |  | ||||||
|   Windows-Taste und dann das Zahnrad um die Einstellungen zu öffnen. Dort wählen |  | ||||||
|   Sie "System", dann "Info" (links unten) und nun |  | ||||||
|   "Erweiterte Systemeinstellungen" (rechts) um den Dialog "Systemeigenschaften" |  | ||||||
|   zu starten. Im Reiter "Erweitert" klicken Sie |  | ||||||
|   "Umgebungsvariablen..." und klicken dann unter "Benutzervariablen" den Knopf |  | ||||||
|   "Neu..." um JAVA_HOME anzulegen oder "Bearbeiten" um ihn zu ändern. Geben Sie |  | ||||||
|   als Name `JAVA_HOME` und als Wert den Pfad ein. Schließen Sie mit "OK". |  | ||||||
|  |  | ||||||
|   > **(!) Beachten Sie, dass Sie die jeweilige Applikation neu starten müssen**, |  | ||||||
|   > um von der gesetzten Umgebungsvariablen Notiz zu nehmen. |  | ||||||
|   > Dies betrifft auch die Shell, die Sie gerade verwenden. |  | ||||||
|  |  | ||||||
| * **UNIX (Linux/MacOS):** |  | ||||||
|  |  | ||||||
|   Öffnen oder erstellen Sie die Datei `~/.profile` (wenn Sie die Bash verwenden; |  | ||||||
|   bei anderen Shells sind es andere Dateien) und ergänzen Sie am Ende der Datei |  | ||||||
|   die Zeile: |  | ||||||
|  |  | ||||||
|   `export JAVA_HOME="<Pfad zum entpackten Archiv>"` |  | ||||||
|  |  | ||||||
|   Ersetzen Sie dabei `<Pfad zum entpackten Archiv>` mit dem entsprechenden Pfad. |  | ||||||
|   Zum Beispiel: |  | ||||||
|  |  | ||||||
|   `export JAVA_HOME="/home/user/jdk-20.0.2"` |  | ||||||
|  |  | ||||||
|   Fügen Sie dann die folgende Zeile hinzu: |  | ||||||
|  |  | ||||||
|   `export PATH="$JAVA_HOME/bin:$PATH"` |  | ||||||
|  |  | ||||||
| ## 2 Programmstart |  | ||||||
|  |  | ||||||
| Grundsätzlich kann man das gesamte Projekt einfach in IntelliJ öffnen. Details |  | ||||||
| dazu sind im Aufgabenblatt zur Einarbeitungsaufgabe zu finden. Im Folgenden ist |  | ||||||
| beschrieben, wie die _Batttleships_ unmittelbar von der Kommandozeile gestartet |  | ||||||
| werden können. |  | ||||||
|  |  | ||||||
| Um _Battleships_ spielen zu können, muss man zuerst das Server-Programm auf |  | ||||||
| einem Rechner und dann zweimal das Client-Programm auf beliebigen Rechnern |  | ||||||
| starten, die TCP/IP-Verbindungen zum Server erlauben. Natürlich ist es auch |  | ||||||
| möglich, alle drei Programme auf demselben Rechner zu starten. |  | ||||||
|  |  | ||||||
| Es empfiehlt sich der Start von der Kommandozeile. Will man alle drei Programme |  | ||||||
| auf demselben Rechner starten, sollte man dazu drei Shell-Instanzen öffnen und |  | ||||||
| in jeder eines der Programme starten. Auf diese Weise können die |  | ||||||
| Logging-Ausgaben der drei Programme voneinander unterschieden werden. |  | ||||||
|  |  | ||||||
| Das Server-Programm startet man unmittelbar mit Gradle mit |  | ||||||
|  |  | ||||||
| `./gradlew :battleship:server:run` |  | ||||||
|  |  | ||||||
| Unter Windows kann es je nach Shell (Eingabeaufforderung cmd) erforderlich sein, |  | ||||||
| `/` jeweils durch `\ ` zu ersetzen. |  | ||||||
|  |  | ||||||
| Im Verzeichnis `battleship/server` befindet sich die Datei `config.propeties`, |  | ||||||
| worüber sich der Server konfigurieren lässt. Mit der Zeile `port=1234` lässt |  | ||||||
| sich der verwendete Server-Port (hier 1234) einstellen. Außerdem befindet sich |  | ||||||
| dort die Datei `logging.properties`, womit das Logging des Servers konfiguriert |  | ||||||
| wird. |  | ||||||
|  |  | ||||||
| Das Client-Programm startet man unmittelbar mit Gradle mit |  | ||||||
|  |  | ||||||
| `./gradlew :battleship:client:run` |  | ||||||
|  |  | ||||||
| Die Datei `logging.properties` im Verzeichnis `battleship/client` konfiguriert |  | ||||||
| das Logging des Clients. |  | ||||||
|  |  | ||||||
| Alternativ kann man auch die Start-Skripte |  | ||||||
|  |  | ||||||
| * `./battleship/server/build/install/battleship-server/bin/battleship-server` |  | ||||||
| * `./battleship/client/build/install/battleship/bin/battleship` |  | ||||||
|  |  | ||||||
| direkt in der Kommandozeile starten. Allerdings müssen sie zuvor mittels |  | ||||||
|  |  | ||||||
| `./gradlew installDist` |  | ||||||
|  |  | ||||||
| erzeugt worden sein. Beachten Sie aber, dass nur im **aktuellen |  | ||||||
| Arbeitsverzeichnis** nach den Dateien `config.properties` und |  | ||||||
| `logging.properties` gesucht wird und diese geladen werden. Das heißt, dass die |  | ||||||
| vordefinierten Dateien in den Verzeichnissen `battleship/server` und |  | ||||||
| `battleship/client` nur dann gelesen werden, wenn Sie diese Verzeichnisse als |  | ||||||
| aktuelle Arbeitsverzeichnisse nutzen. Wie üblich müssen Sie dazu in der |  | ||||||
| Kommandozeile |  | ||||||
|  |  | ||||||
| `cd battleship/server` |  | ||||||
|  |  | ||||||
| bzw. |  | ||||||
|  |  | ||||||
| `cd battleship/client` |  | ||||||
|  |  | ||||||
| eingeben. |  | ||||||
|  |  | ||||||
| ## 3 Hinweise zu _Battleships_ |  | ||||||
|  |  | ||||||
| Der _Battleships_-Client hat ein Menü, in das man immer mit der |  | ||||||
| Esc-Taste kommt. Aus dem Menü heraus lässt sich das Programm auch schließen. |  | ||||||
| Beachte, dass sich beim Laden und Speichern eines Spiels kein Dateidialog |  | ||||||
| öffnet. Vielmehr muss man den Dateipfad in das Dialogfeld eingeben. Da |  | ||||||
| JSON-Dateien geschrieben werden, empfiehlt sich das Datei-Suffix _.json_. |  | ||||||
|  |  | ||||||
| ## 4 Allgemeine Gradle-Tasks: |  | ||||||
|  |  | ||||||
| - `./gradlew clean` |  | ||||||
|  |  | ||||||
|   Entfernt alle `build`-Verzeichnisse und alle erzeugten Dateien. |  | ||||||
|  |  | ||||||
| - `./gradlew classes` |  | ||||||
|  |  | ||||||
|   Übersetzt den Quellcode und legt unter build den Bytecode sowie |  | ||||||
|   Ressourcen ab. |  | ||||||
|  |  | ||||||
| - `./gradlew javadoc` |  | ||||||
|  |  | ||||||
|   Erzeugt die Dokumentation aus den JavaDoc-Kommentaren im Verzeichnis |  | ||||||
|   `build/docs/javadoc` des jeweiligen Unterprojekts. |  | ||||||
|  |  | ||||||
| - `./gradlew test` |  | ||||||
|  |  | ||||||
|   Führt die JUnit-Tests durch. Ergebnisse sind im Verzeichnis |  | ||||||
|   `build/reports/tests` des jeweiligen Unterprojekts zu finden. |  | ||||||
|  |  | ||||||
| - `./gradlew build` |  | ||||||
|  |  | ||||||
|   Führt die JUnit-Tests durch und erstellt in `build/distributions` |  | ||||||
|   gepackte Distributionsdateien |  | ||||||
|  |  | ||||||
| - `./gradlew installDist` |  | ||||||
|  |  | ||||||
|   Erstellt unter `battleship/client/build/install` und |  | ||||||
|   `battleship/server/build/install` Verzeichnisse, die jeweils eine ausführbare |  | ||||||
|   Distribution samt Start-Skripten enthält (siehe oben). |  | ||||||
|  |  | ||||||
| --- |  | ||||||
| Juli 2024 |  | ||||||
| @@ -1,22 +0,0 @@ | |||||||
| plugins { |  | ||||||
|     id 'buildlogic.jme-application-conventions' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| description = 'Battleship Client' |  | ||||||
|  |  | ||||||
| dependencies { |  | ||||||
|     implementation project(":jme-common") |  | ||||||
|     implementation project(":battleship:model") |  | ||||||
|  |  | ||||||
|     implementation libs.jme3.desktop |  | ||||||
|  |  | ||||||
|     runtimeOnly libs.jme3.awt.dialogs |  | ||||||
|     runtimeOnly libs.jme3.plugins |  | ||||||
|     runtimeOnly libs.jme3.jogg |  | ||||||
|     runtimeOnly libs.jme3.testdata |  | ||||||
| } |  | ||||||
|  |  | ||||||
| application { |  | ||||||
|     mainClass = 'pp.battleship.client.BattleshipApp' |  | ||||||
|     applicationName = 'battleship' |  | ||||||
| } |  | ||||||
| @@ -1,73 +0,0 @@ | |||||||
| ######################################## |  | ||||||
| ## Programming project code |  | ||||||
| ## UniBw M, 2022, 2023, 2024 |  | ||||||
| ## www.unibw.de/inf2 |  | ||||||
| ## (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| ######################################## |  | ||||||
| # |  | ||||||
| # Battleship client configuration |  | ||||||
| # |  | ||||||
| # Specifies the map used by the opponent in single mode. |  | ||||||
| # Single mode is activated if this property is set. |  | ||||||
| #map.opponent=maps/map2.json |  | ||||||
| # |  | ||||||
| # Specifies the map used by the player in single mode. |  | ||||||
| # The player must define their own map if this property is not set. |  | ||||||
| map.own=maps/map1.json |  | ||||||
| # |  | ||||||
| # Coordinates of the shots fired by the RobotClient in the order listed. |  | ||||||
| # Example: |  | ||||||
| #   2, 0,\ |  | ||||||
| #   2, 1,\ |  | ||||||
| #   2, 2,\ |  | ||||||
| #   2, 3 |  | ||||||
| #  defines four shots, namely at the coordinates |  | ||||||
| #  (x=2, y=0), (x=2, y=1), (x=2, y=2), and (x=2, y=3) |  | ||||||
| robot.targets=2, 0,\ |  | ||||||
|               2, 1,\ |  | ||||||
|               2, 2,\ |  | ||||||
|               2, 3 |  | ||||||
| # |  | ||||||
| # Delay in milliseconds between each shot fired by the RobotClient. |  | ||||||
| robot.delay=500 |  | ||||||
| # |  | ||||||
| # The dimensions of the game map used in single mode. |  | ||||||
| # 'map.width' defines the number of columns, and 'map.height' defines the number of rows. |  | ||||||
| map.width=10 |  | ||||||
| map.height=10 |  | ||||||
| # |  | ||||||
| # The number of ships of each length available in single mode. |  | ||||||
| # The value is a comma-separated list where each element corresponds to the number of ships |  | ||||||
| # with a specific length. For example: |  | ||||||
| # ship.nums=4, 3, 2, 1 |  | ||||||
| # This configuration means: |  | ||||||
| #   - 4 ships of length 1 |  | ||||||
| #   - 3 ships of length 2 |  | ||||||
| #   - 2 ships of length 3 |  | ||||||
| #   - 1 ship of length 4 |  | ||||||
| ship.nums=4, 3, 2, 1 |  | ||||||
| # |  | ||||||
| # Screen settings |  | ||||||
| # |  | ||||||
| # Color of the text displayed at the top of the overlay. |  | ||||||
| # The format is (red, green, blue, alpha) where each value ranges from 0 to 1. |  | ||||||
| overlay.top.color=1, 1, 1, 1 |  | ||||||
| # |  | ||||||
| # Application settings configuration |  | ||||||
| # Determines whether the settings window is shown at startup. |  | ||||||
| settings.show=false |  | ||||||
| # |  | ||||||
| # Specifies the width of the application window in pixels. |  | ||||||
| settings.resolution.width=1200 |  | ||||||
| # |  | ||||||
| # Specifies the height of the application window in pixels. |  | ||||||
| settings.resolution.height=800 |  | ||||||
| # |  | ||||||
| # Determines whether the application runs in full-screen mode. |  | ||||||
| settings.full-screen=false |  | ||||||
| # |  | ||||||
| # Enables or disables gamma correction to improve color accuracy. |  | ||||||
| settings.use-gamma-correction=true |  | ||||||
| # |  | ||||||
| # Indicates whether the statistics window is displayed during gameplay. |  | ||||||
| statistics.show=false |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| handlers=java.util.logging.ConsoleHandler |  | ||||||
| .level=INFO |  | ||||||
| pp.level=FINE |  | ||||||
| com.jme3.network.level=INFO |  | ||||||
| ;com.jme3.util.TangentBinormalGenerator.level=SEVERE |  | ||||||
| java.util.logging.ConsoleHandler.level=FINER |  | ||||||
| java.util.logging.ConsoleHandler.formatter=java.util.logging.SimpleFormatter |  | ||||||
| ;java.util.logging.SimpleFormatter.format=[%4$s %2$s] %5$s%n |  | ||||||
| @@ -1,66 +0,0 @@ | |||||||
| { |  | ||||||
|   "width": 10, |  | ||||||
|   "height": 10, |  | ||||||
|   "ships": [ |  | ||||||
|     { |  | ||||||
|       "length": 4, |  | ||||||
|       "x": 2, |  | ||||||
|       "y": 8, |  | ||||||
|       "rot": "RIGHT" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "length": 3, |  | ||||||
|       "x": 2, |  | ||||||
|       "y": 5, |  | ||||||
|       "rot": "DOWN" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "length": 3, |  | ||||||
|       "x": 5, |  | ||||||
|       "y": 6, |  | ||||||
|       "rot": "RIGHT" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "length": 2, |  | ||||||
|       "x": 4, |  | ||||||
|       "y": 4, |  | ||||||
|       "rot": "RIGHT" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "length": 2, |  | ||||||
|       "x": 7, |  | ||||||
|       "y": 4, |  | ||||||
|       "rot": "RIGHT" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "length": 2, |  | ||||||
|       "x": 4, |  | ||||||
|       "y": 2, |  | ||||||
|       "rot": "DOWN" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "length": 1, |  | ||||||
|       "x": 6, |  | ||||||
|       "y": 2, |  | ||||||
|       "rot": "RIGHT" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "length": 1, |  | ||||||
|       "x": 8, |  | ||||||
|       "y": 2, |  | ||||||
|       "rot": "RIGHT" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "length": 1, |  | ||||||
|       "x": 6, |  | ||||||
|       "y": 0, |  | ||||||
|       "rot": "RIGHT" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "length": 1, |  | ||||||
|       "x": 8, |  | ||||||
|       "y": 0, |  | ||||||
|       "rot": "RIGHT" |  | ||||||
|     } |  | ||||||
|   ] |  | ||||||
| } |  | ||||||
| @@ -1,66 +0,0 @@ | |||||||
| { |  | ||||||
|   "width": 10, |  | ||||||
|   "height": 10, |  | ||||||
|   "ships": [ |  | ||||||
|     { |  | ||||||
|       "length": 4, |  | ||||||
|       "x": 0, |  | ||||||
|       "y": 5, |  | ||||||
|       "rot": "DOWN" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "length": 3, |  | ||||||
|       "x": 0, |  | ||||||
|       "y": 9, |  | ||||||
|       "rot": "DOWN" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "length": 3, |  | ||||||
|       "x": 2, |  | ||||||
|       "y": 6, |  | ||||||
|       "rot": "RIGHT" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "length": 2, |  | ||||||
|       "x": 4, |  | ||||||
|       "y": 8, |  | ||||||
|       "rot": "RIGHT" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "length": 2, |  | ||||||
|       "x": 2, |  | ||||||
|       "y": 4, |  | ||||||
|       "rot": "DOWN" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "length": 2, |  | ||||||
|       "x": 2, |  | ||||||
|       "y": 1, |  | ||||||
|       "rot": "DOWN" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "length": 1, |  | ||||||
|       "x": 6, |  | ||||||
|       "y": 2, |  | ||||||
|       "rot": "RIGHT" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "length": 1, |  | ||||||
|       "x": 8, |  | ||||||
|       "y": 2, |  | ||||||
|       "rot": "RIGHT" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "length": 1, |  | ||||||
|       "x": 6, |  | ||||||
|       "y": 0, |  | ||||||
|       "rot": "RIGHT" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "length": 1, |  | ||||||
|       "x": 8, |  | ||||||
|       "y": 0, |  | ||||||
|       "rot": "RIGHT" |  | ||||||
|     } |  | ||||||
|   ] |  | ||||||
| } |  | ||||||
| @@ -1,429 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.client; |  | ||||||
|  |  | ||||||
| import com.jme3.app.DebugKeysAppState; |  | ||||||
| import com.jme3.app.SimpleApplication; |  | ||||||
| import com.jme3.app.StatsAppState; |  | ||||||
| import com.jme3.font.BitmapFont; |  | ||||||
| import com.jme3.font.BitmapText; |  | ||||||
| import com.jme3.input.KeyInput; |  | ||||||
| import com.jme3.input.MouseInput; |  | ||||||
| import com.jme3.input.controls.ActionListener; |  | ||||||
| import com.jme3.input.controls.KeyTrigger; |  | ||||||
| import com.jme3.input.controls.MouseButtonTrigger; |  | ||||||
| import com.jme3.system.AppSettings; |  | ||||||
| import com.simsilica.lemur.GuiGlobals; |  | ||||||
| import com.simsilica.lemur.style.BaseStyles; |  | ||||||
| import pp.battleship.client.gui.BattleAppState; |  | ||||||
| import pp.battleship.client.gui.EditorAppState; |  | ||||||
| import pp.battleship.client.gui.SeaAppState; |  | ||||||
| import pp.battleship.game.client.BattleshipClient; |  | ||||||
| import pp.battleship.game.client.ClientGameLogic; |  | ||||||
| import pp.battleship.game.client.ServerConnection; |  | ||||||
| import pp.battleship.game.singlemode.BattleshipClientConfig; |  | ||||||
| import pp.battleship.game.singlemode.ServerConnectionMockup; |  | ||||||
| import pp.battleship.notification.ClientStateEvent; |  | ||||||
| import pp.battleship.notification.GameEventListener; |  | ||||||
| import pp.battleship.notification.InfoTextEvent; |  | ||||||
| import pp.dialog.DialogBuilder; |  | ||||||
| import pp.dialog.DialogManager; |  | ||||||
| import pp.graphics.Draw; |  | ||||||
|  |  | ||||||
| import java.io.File; |  | ||||||
| import java.io.FileInputStream; |  | ||||||
| import java.io.IOException; |  | ||||||
| import java.lang.System.Logger; |  | ||||||
| import java.lang.System.Logger.Level; |  | ||||||
| import java.util.concurrent.ExecutorService; |  | ||||||
| import java.util.concurrent.Executors; |  | ||||||
| import java.util.logging.LogManager; |  | ||||||
|  |  | ||||||
| import static pp.battleship.Resources.lookup; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * The main class for the Battleship client application. |  | ||||||
|  * It manages the initialization, input setup, GUI setup, and game states for the client. |  | ||||||
|  */ |  | ||||||
| public class BattleshipApp extends SimpleApplication implements BattleshipClient, GameEventListener { |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Logger for logging messages within the application. |  | ||||||
|      */ |  | ||||||
|     private static final Logger LOGGER = System.getLogger(BattleshipApp.class.getName()); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Path to the styles script for GUI elements. |  | ||||||
|      */ |  | ||||||
|     private static final String STYLES_SCRIPT = "Interface/Lemur/pp-styles.groovy"; //NON-NLS |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Path to the font resource used in the GUI. |  | ||||||
|      */ |  | ||||||
|     private static final String FONT = "Interface/Fonts/Default.fnt"; //NON-NLS |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Path to the client configuration file, if one exists. |  | ||||||
|      */ |  | ||||||
|     private static final File CONFIG_FILE = new File("client.properties"); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Input mapping name for mouse clicks. |  | ||||||
|      */ |  | ||||||
|     public static final String CLICK = "CLICK"; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Input mapping name for the Escape key. |  | ||||||
|      */ |  | ||||||
|     private static final String ESC = "ESC"; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Manager for handling dialogs within the application. |  | ||||||
|      */ |  | ||||||
|     private final DialogManager dialogManager = new DialogManager(this); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The server connection instance, used for communicating with the game server. |  | ||||||
|      */ |  | ||||||
|     private final ServerConnection serverConnection; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Instance of the {@link Draw} class for rendering graphics. |  | ||||||
|      */ |  | ||||||
|     private Draw draw; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Text display at the top of the GUI for showing information to the user. |  | ||||||
|      */ |  | ||||||
|     private BitmapText topText; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Executor service for handling asynchronous tasks within the application. |  | ||||||
|      */ |  | ||||||
|     private ExecutorService executor; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handler for managing the client's game logic. |  | ||||||
|      */ |  | ||||||
|     private final ClientGameLogic logic; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Configuration settings for the Battleship client application. |  | ||||||
|      */ |  | ||||||
|     private final BattleshipAppConfig config; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Listener for handling actions triggered by the Escape key. |  | ||||||
|      */ |  | ||||||
|     private final ActionListener escapeListener = (name, isPressed, tpf) -> escape(isPressed); |  | ||||||
|  |  | ||||||
|     static { |  | ||||||
|         // Configure logging |  | ||||||
|         LogManager manager = LogManager.getLogManager(); |  | ||||||
|         try { |  | ||||||
|             manager.readConfiguration(new FileInputStream("logging.properties")); |  | ||||||
|             LOGGER.log(Level.INFO, "Successfully read logging properties"); //NON-NLS |  | ||||||
|         } |  | ||||||
|         catch (IOException e) { |  | ||||||
|             LOGGER.log(Level.INFO, e.getMessage()); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Starts the Battleship application. |  | ||||||
|      * |  | ||||||
|      * @param args Command-line arguments for launching the application. |  | ||||||
|      */ |  | ||||||
|     public static void main(String[] args) { |  | ||||||
|         new BattleshipApp().start(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a new {@code BattleshipApp} instance. |  | ||||||
|      * Initializes the configuration, server connection, and game logic listeners. |  | ||||||
|      */ |  | ||||||
|     private BattleshipApp() { |  | ||||||
|         config = new BattleshipAppConfig(); |  | ||||||
|         config.readFromIfExists(CONFIG_FILE); |  | ||||||
|         serverConnection = makeServerConnection(); |  | ||||||
|         logic = new ClientGameLogic(serverConnection); |  | ||||||
|         logic.addListener(this); |  | ||||||
|         setShowSettings(config.getShowSettings()); |  | ||||||
|         setSettings(makeSettings()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates and configures application settings from the client configuration. |  | ||||||
|      * |  | ||||||
|      * @return A configured {@link AppSettings} object. |  | ||||||
|      */ |  | ||||||
|     private AppSettings makeSettings() { |  | ||||||
|         final AppSettings settings = new AppSettings(true); |  | ||||||
|         settings.setTitle(lookup("battleship.name")); |  | ||||||
|         settings.setResolution(config.getResolutionWidth(), config.getResolutionHeight()); |  | ||||||
|         settings.setFullscreen(config.fullScreen()); |  | ||||||
|         settings.setUseRetinaFrameBuffer(config.useRetinaFrameBuffer()); |  | ||||||
|         settings.setGammaCorrection(config.useGammaCorrection()); |  | ||||||
|         return settings; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Factory method for creating a server connection based on the current |  | ||||||
|      * client configuration. |  | ||||||
|      * |  | ||||||
|      * @return A {@link ServerConnection} instance, which could be a real or mock server. |  | ||||||
|      */ |  | ||||||
|     private ServerConnection makeServerConnection() { |  | ||||||
|         if (config.isSingleMode()) |  | ||||||
|             return new ServerConnectionMockup(this); |  | ||||||
|         return new NetworkSupport(this); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the dialog manager responsible for managing in-game dialogs. |  | ||||||
|      * |  | ||||||
|      * @return The {@link DialogManager} instance. |  | ||||||
|      */ |  | ||||||
|     DialogManager getDialogManager() { |  | ||||||
|         return dialogManager; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the game logic handler for the client. |  | ||||||
|      * |  | ||||||
|      * @return The {@link ClientGameLogic} instance. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public ClientGameLogic getGameLogic() { |  | ||||||
|         return logic; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the current configuration settings for the Battleship client. |  | ||||||
|      * |  | ||||||
|      * @return The {@link BattleshipClientConfig} instance. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public BattleshipAppConfig getConfig() { |  | ||||||
|         return config; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Initializes the application. |  | ||||||
|      * Sets up input mappings, GUI, game states, and connects to the server. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void simpleInitApp() { |  | ||||||
|         setPauseOnLostFocus(false); |  | ||||||
|         draw = new Draw(assetManager); |  | ||||||
|         setupInput(); |  | ||||||
|         setupStates(); |  | ||||||
|         setupGui(); |  | ||||||
|         serverConnection.connect(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Sets up the graphical user interface (GUI) for the application. |  | ||||||
|      */ |  | ||||||
|     private void setupGui() { |  | ||||||
|         GuiGlobals.initialize(this); |  | ||||||
|         BaseStyles.loadStyleResources(STYLES_SCRIPT); |  | ||||||
|         GuiGlobals.getInstance().getStyles().setDefaultStyle("pp"); //NON-NLS |  | ||||||
|         final BitmapFont normalFont = assetManager.loadFont(FONT); //NON-NLS |  | ||||||
|         topText = new BitmapText(normalFont); |  | ||||||
|         final int height = context.getSettings().getHeight(); |  | ||||||
|         topText.setLocalTranslation(10f, height - 10f, 0f); |  | ||||||
|         topText.setColor(config.getTopColor()); |  | ||||||
|         guiNode.attachChild(topText); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Configures input mappings and sets up listeners for user interactions. |  | ||||||
|      */ |  | ||||||
|     private void setupInput() { |  | ||||||
|         inputManager.deleteMapping(INPUT_MAPPING_EXIT); |  | ||||||
|         inputManager.setCursorVisible(false); |  | ||||||
|         inputManager.addMapping(ESC, new KeyTrigger(KeyInput.KEY_ESCAPE)); |  | ||||||
|         inputManager.addMapping(CLICK, new MouseButtonTrigger(MouseInput.BUTTON_LEFT)); |  | ||||||
|         inputManager.addListener(escapeListener, ESC); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Initializes and attaches the necessary application states for the game. |  | ||||||
|      */ |  | ||||||
|     private void setupStates() { |  | ||||||
|         if (config.getShowStatistics()) { |  | ||||||
|             final BitmapFont normalFont = assetManager.loadFont(FONT); //NON-NLS |  | ||||||
|             final StatsAppState stats = new StatsAppState(guiNode, normalFont); |  | ||||||
|             stateManager.attach(stats); |  | ||||||
|         } |  | ||||||
|         flyCam.setEnabled(false); |  | ||||||
|         stateManager.detach(stateManager.getState(StatsAppState.class)); |  | ||||||
|         stateManager.detach(stateManager.getState(DebugKeysAppState.class)); |  | ||||||
|  |  | ||||||
|         attachGameSound(); |  | ||||||
|         stateManager.attachAll(new EditorAppState(), new BattleAppState(), new SeaAppState()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Attaches the game sound state and sets its initial enabled state. |  | ||||||
|      */ |  | ||||||
|     private void attachGameSound() { |  | ||||||
|         final GameSound gameSound = new GameSound(); |  | ||||||
|         logic.addListener(gameSound); |  | ||||||
|         gameSound.setEnabled(GameSound.enabledInPreferences()); |  | ||||||
|         stateManager.attach(gameSound); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Updates the application state every frame. |  | ||||||
|      * This method is called once per frame during the game loop. |  | ||||||
|      * |  | ||||||
|      * @param tpf Time per frame in seconds. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void simpleUpdate(float tpf) { |  | ||||||
|         super.simpleUpdate(tpf); |  | ||||||
|         dialogManager.update(tpf); |  | ||||||
|         logic.update(tpf); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles the Escape key action to either close the top dialog or show the main menu. |  | ||||||
|      * |  | ||||||
|      * @param isPressed Indicates whether the Escape key is pressed. |  | ||||||
|      */ |  | ||||||
|     private void escape(boolean isPressed) { |  | ||||||
|         if (!isPressed) return; |  | ||||||
|         if (dialogManager.showsDialog()) |  | ||||||
|             dialogManager.escape(); |  | ||||||
|         else |  | ||||||
|             new Menu(this).open(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the {@link Draw} instance used for rendering graphical elements in the game. |  | ||||||
|      * |  | ||||||
|      * @return The {@link Draw} instance. |  | ||||||
|      */ |  | ||||||
|     public Draw getDraw() { |  | ||||||
|         return draw; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles a request to close the application. |  | ||||||
|      * If the request is initiated by pressing ESC, this parameter is true. |  | ||||||
|      * |  | ||||||
|      * @param esc If true, the request is due to the ESC key being pressed. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void requestClose(boolean esc) { /* do nothing */ } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Closes the application, displaying a confirmation dialog if the client is connected to a server. |  | ||||||
|      */ |  | ||||||
|     public void closeApp() { |  | ||||||
|         if (serverConnection.isConnected()) |  | ||||||
|             confirmDialog(lookup("confirm.leaving"), this::close); |  | ||||||
|         else |  | ||||||
|             close(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Closes the application, disconnecting from the server and stopping the application. |  | ||||||
|      */ |  | ||||||
|     private void close() { |  | ||||||
|         serverConnection.disconnect(); |  | ||||||
|         stop(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Updates the informational text displayed in the GUI. |  | ||||||
|      * |  | ||||||
|      * @param text The information text to display. |  | ||||||
|      */ |  | ||||||
|     void setInfoText(String text) { |  | ||||||
|         LOGGER.log(Level.DEBUG, "setInfoText {0}", text); //NON-NLS |  | ||||||
|         topText.setText(text); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Updates the informational text in the GUI based on the key received in an {@link InfoTextEvent}. |  | ||||||
|      * |  | ||||||
|      * @param event The {@link InfoTextEvent} containing the key for the text to display. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void receivedEvent(InfoTextEvent event) { |  | ||||||
|         LOGGER.log(Level.DEBUG, "received info text {0}", event.key()); //NON-NLS |  | ||||||
|         setInfoText(lookup(event.key())); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles client state events to update the game states accordingly. |  | ||||||
|      * |  | ||||||
|      * @param event The {@link ClientStateEvent} representing the state change. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void receivedEvent(ClientStateEvent event) { |  | ||||||
|         stateManager.getState(EditorAppState.class).setEnabled(logic.showEditor()); |  | ||||||
|         stateManager.getState(BattleAppState.class).setEnabled(logic.showBattle()); |  | ||||||
|         stateManager.getState(SeaAppState.class).setEnabled(logic.showBattle()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the executor service used for handling multithreaded tasks. |  | ||||||
|      * |  | ||||||
|      * @return The {@link ExecutorService} instance. |  | ||||||
|      */ |  | ||||||
|     public ExecutorService getExecutor() { |  | ||||||
|         if (executor == null) |  | ||||||
|             executor = Executors.newCachedThreadPool(); |  | ||||||
|         return executor; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Stops the application, shutting down the executor service and halting execution. |  | ||||||
|      * |  | ||||||
|      * @param waitFor If true, waits for the application to stop before returning. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void stop(boolean waitFor) { |  | ||||||
|         if (executor != null) executor.shutdownNow(); |  | ||||||
|         super.stop(waitFor); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Displays a confirmation dialog with a specified question and action for the "Yes" button. |  | ||||||
|      * |  | ||||||
|      * @param question  The question to display in the dialog. |  | ||||||
|      * @param yesAction The action to perform if "Yes" is selected. |  | ||||||
|      */ |  | ||||||
|     void confirmDialog(String question, Runnable yesAction) { |  | ||||||
|         DialogBuilder.simple(dialogManager) |  | ||||||
|                      .setTitle(lookup("dialog.question")) |  | ||||||
|                      .setText(question) |  | ||||||
|                      .setOkButton(lookup("button.yes"), yesAction) |  | ||||||
|                      .setNoButton(lookup("button.no")) |  | ||||||
|                      .build() |  | ||||||
|                      .open(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Displays an error dialog with the specified error message. |  | ||||||
|      * |  | ||||||
|      * @param errorMessage The error message to display in the dialog. |  | ||||||
|      */ |  | ||||||
|     void errorDialog(String errorMessage) { |  | ||||||
|         DialogBuilder.simple(dialogManager) |  | ||||||
|                      .setTitle(lookup("dialog.error")) |  | ||||||
|                      .setText(errorMessage) |  | ||||||
|                      .setOkButton(lookup("button.ok")) |  | ||||||
|                      .build() |  | ||||||
|                      .open(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,196 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.client; |  | ||||||
|  |  | ||||||
| import com.jme3.math.ColorRGBA; |  | ||||||
| import pp.battleship.game.singlemode.BattleshipClientConfig; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Provides access to the Battleship application configuration. |  | ||||||
|  * Extends {@link BattleshipClientConfig} to include additional properties specific to the client, |  | ||||||
|  * particularly those related to screen settings and visual customization. |  | ||||||
|  * <p> |  | ||||||
|  * <b>Note:</b> Attributes of this class should not be marked as {@code final} |  | ||||||
|  * to ensure proper functionality when reading from a properties file. |  | ||||||
|  * </p> |  | ||||||
|  */ |  | ||||||
| public class BattleshipAppConfig extends BattleshipClientConfig { |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Converts a string value found in the properties file into an object of the specified type. |  | ||||||
|      * Extends the superclass method to support conversion to {@link ColorRGBA}. |  | ||||||
|      * |  | ||||||
|      * @param value      the string value to be converted |  | ||||||
|      * @param targetType the target type into which the value string is converted |  | ||||||
|      * @return the converted object of the specified type |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     protected Object convertToType(String value, Class<?> targetType) { |  | ||||||
|         if (targetType == ColorRGBA.class) |  | ||||||
|             return makeColorRGBA(value); |  | ||||||
|         return super.convertToType(value, targetType); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Converts the specified string value to a corresponding {@link ColorRGBA} object. |  | ||||||
|      * |  | ||||||
|      * @param value the color in the format "red, green, blue, alpha" with all values in the range [0..1] |  | ||||||
|      * @return a {@link ColorRGBA} object representing the color |  | ||||||
|      * @throws IllegalArgumentException if the input string is not in the expected format |  | ||||||
|      */ |  | ||||||
|     private static ColorRGBA makeColorRGBA(String value) { |  | ||||||
|         String[] split = value.split(",", -1); |  | ||||||
|         try { |  | ||||||
|             if (split.length == 4) |  | ||||||
|                 return new ColorRGBA(Float.parseFloat(split[0]), |  | ||||||
|                                      Float.parseFloat(split[1]), |  | ||||||
|                                      Float.parseFloat(split[2]), |  | ||||||
|                                      Float.parseFloat(split[3])); |  | ||||||
|         } |  | ||||||
|         catch (NumberFormatException e) { |  | ||||||
|             // deliberately left empty |  | ||||||
|         } |  | ||||||
|         throw new IllegalArgumentException(value + " should consist of exactly 4 numbers"); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The width of the game view resolution in pixels. |  | ||||||
|      */ |  | ||||||
|     @Property("settings.resolution.width") //NON-NLS |  | ||||||
|     private int resolutionWidth = 1200; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The height of the game view resolution in pixels. |  | ||||||
|      */ |  | ||||||
|     @Property("settings.resolution.height") //NON-NLS |  | ||||||
|     private int resolutionHeight = 800; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Specifies whether the game should start in full-screen mode. |  | ||||||
|      */ |  | ||||||
|     @Property("settings.full-screen") //NON-NLS |  | ||||||
|     private boolean fullScreen = false; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Specifies whether gamma correction should be enabled. |  | ||||||
|      * If enabled, the main framebuffer is configured for sRGB colors, |  | ||||||
|      * and sRGB images are linearized. |  | ||||||
|      * <p> |  | ||||||
|      * Requires a GPU that supports GL_ARB_framebuffer_sRGB; otherwise, this setting will be ignored. |  | ||||||
|      * </p> |  | ||||||
|      */ |  | ||||||
|     @Property("settings.use-gamma-correction") //NON-NLS |  | ||||||
|     private boolean useGammaCorrection = true; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Specifies whether full resolution framebuffers should be used on Retina displays. |  | ||||||
|      * This setting is ignored on non-Retina platforms. |  | ||||||
|      */ |  | ||||||
|     @Property("settings.use-retina-framebuffer") //NON-NLS |  | ||||||
|     private boolean useRetinaFrameBuffer = false; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Specifies whether the settings window should be shown for configuring the game. |  | ||||||
|      */ |  | ||||||
|     @Property("settings.show") //NON-NLS |  | ||||||
|     private boolean showSettings = false; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Specifies whether the JME statistics window should be shown in the lower left corner of the screen. |  | ||||||
|      */ |  | ||||||
|     @Property("statistics.show") //NON-NLS |  | ||||||
|     private boolean showStatistics = false; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The color of the top text during gameplay, represented as a {@link ColorRGBA} object. |  | ||||||
|      */ |  | ||||||
|     @Property("overlay.top.color") //NON-NLS |  | ||||||
|     private ColorRGBA topColor = ColorRGBA.White; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates a default {@code BattleshipAppConfig} with predefined values. |  | ||||||
|      */ |  | ||||||
|     public BattleshipAppConfig() { |  | ||||||
|         // Default constructor |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the width of the game view resolution in pixels. |  | ||||||
|      * |  | ||||||
|      * @return the width of the game view resolution in pixels |  | ||||||
|      */ |  | ||||||
|     public int getResolutionWidth() { |  | ||||||
|         return resolutionWidth; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the height of the game view resolution in pixels. |  | ||||||
|      * |  | ||||||
|      * @return the height of the game view resolution in pixels |  | ||||||
|      */ |  | ||||||
|     public int getResolutionHeight() { |  | ||||||
|         return resolutionHeight; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns whether the game should start in full-screen mode. |  | ||||||
|      * |  | ||||||
|      * @return {@code true} if the game should start in full-screen mode; {@code false} otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean fullScreen() { |  | ||||||
|         return fullScreen; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns whether gamma correction is enabled. |  | ||||||
|      * If enabled, the main framebuffer is configured for sRGB colors, |  | ||||||
|      * and sRGB images are linearized. |  | ||||||
|      * |  | ||||||
|      * @return {@code true} if gamma correction is enabled; {@code false} otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean useGammaCorrection() { |  | ||||||
|         return useGammaCorrection; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns whether full resolution framebuffers should be used on Retina displays. |  | ||||||
|      * This setting is ignored on non-Retina platforms. |  | ||||||
|      * |  | ||||||
|      * @return {@code true} if full resolution framebuffers should be used on Retina displays; {@code false} otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean useRetinaFrameBuffer() { |  | ||||||
|         return useRetinaFrameBuffer; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns whether the settings window should be shown for configuring the game. |  | ||||||
|      * |  | ||||||
|      * @return {@code true} if the settings window should be shown; {@code false} otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean getShowSettings() { |  | ||||||
|         return showSettings; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns whether the JME statistics window should be shown in the lower left corner of the screen. |  | ||||||
|      * |  | ||||||
|      * @return {@code true} if the statistics window should be shown; {@code false} otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean getShowStatistics() { |  | ||||||
|         return showStatistics; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the color of the top text during gameplay as a {@link ColorRGBA} object. |  | ||||||
|      * |  | ||||||
|      * @return the color of the top text during gameplay |  | ||||||
|      */ |  | ||||||
|     public ColorRGBA getTopColor() { |  | ||||||
|         return topColor; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,102 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.client; |  | ||||||
|  |  | ||||||
| import com.jme3.app.Application; |  | ||||||
| import com.jme3.app.state.AbstractAppState; |  | ||||||
| import com.jme3.app.state.AppStateManager; |  | ||||||
| import pp.battleship.game.client.ClientGameLogic; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Abstract class representing a state in the Battleship game. |  | ||||||
|  * Extends the AbstractAppState from jMonkeyEngine to manage state behavior. |  | ||||||
|  */ |  | ||||||
| public abstract class BattleshipAppState extends AbstractAppState { |  | ||||||
|     private BattleshipApp app; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates a new BattleshipAppState that is initially disabled. |  | ||||||
|      * |  | ||||||
|      * @see #setEnabled(boolean) |  | ||||||
|      */ |  | ||||||
|     protected BattleshipAppState() { |  | ||||||
|         setEnabled(false); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Initializes the state manager and application. |  | ||||||
|      * |  | ||||||
|      * @param stateManager The state manager |  | ||||||
|      * @param application  The application instance |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void initialize(AppStateManager stateManager, Application application) { |  | ||||||
|         super.initialize(stateManager, application); |  | ||||||
|         this.app = (BattleshipApp) application; |  | ||||||
|         if (isEnabled()) enableState(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the BattleshipApp instance associated with this BattleshipAppState. |  | ||||||
|      * |  | ||||||
|      * @return The BattleshipApp instance. |  | ||||||
|      */ |  | ||||||
|     public BattleshipApp getApp() { |  | ||||||
|         return app; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the client game logic handler. |  | ||||||
|      * |  | ||||||
|      * @return the client game logic handler |  | ||||||
|      */ |  | ||||||
|     public ClientGameLogic getGameLogic() { |  | ||||||
|         return app.getGameLogic(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if any dialog is currently displayed. |  | ||||||
|      * |  | ||||||
|      * @return true if any dialog is currently shown, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean showsDialog() { |  | ||||||
|         return app.getDialogManager().showsDialog(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Sets the enabled state of the BattleshipAppState. |  | ||||||
|      * If the new state is the same as the current state, the method returns. |  | ||||||
|      * |  | ||||||
|      * @param enabled The new enabled state. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void setEnabled(boolean enabled) { |  | ||||||
|         if (isEnabled() == enabled) return; |  | ||||||
|         super.setEnabled(enabled); |  | ||||||
|         if (app != null) { |  | ||||||
|             if (enabled) |  | ||||||
|                 enableState(); |  | ||||||
|             else |  | ||||||
|                 disableState(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * This method is called when the state is enabled. |  | ||||||
|      * It is meant to be overridden by subclasses to perform |  | ||||||
|      * specific actions when the state is enabled. |  | ||||||
|      */ |  | ||||||
|     protected abstract void enableState(); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * This method is called when the state is disabled. |  | ||||||
|      * It is meant to be overridden by subclasses to perform |  | ||||||
|      * specific actions when the state is disabled. |  | ||||||
|      */ |  | ||||||
|     protected abstract void disableState(); |  | ||||||
| } |  | ||||||
| @@ -1,135 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.client; |  | ||||||
|  |  | ||||||
| import com.jme3.app.Application; |  | ||||||
| import com.jme3.app.state.AbstractAppState; |  | ||||||
| import com.jme3.app.state.AppStateManager; |  | ||||||
| import com.jme3.asset.AssetLoadException; |  | ||||||
| import com.jme3.asset.AssetNotFoundException; |  | ||||||
| import com.jme3.audio.AudioData; |  | ||||||
| import com.jme3.audio.AudioNode; |  | ||||||
| import pp.battleship.notification.GameEventListener; |  | ||||||
| import pp.battleship.notification.SoundEvent; |  | ||||||
|  |  | ||||||
| import java.lang.System.Logger; |  | ||||||
| import java.lang.System.Logger.Level; |  | ||||||
| import java.util.prefs.Preferences; |  | ||||||
|  |  | ||||||
| import static pp.util.PreferencesUtils.getPreferences; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * An application state that plays sounds. |  | ||||||
|  */ |  | ||||||
| public class GameSound extends AbstractAppState implements GameEventListener { |  | ||||||
|     private static final Logger LOGGER = System.getLogger(GameSound.class.getName()); |  | ||||||
|     private static final Preferences PREFERENCES = getPreferences(GameSound.class); |  | ||||||
|     private static final String ENABLED_PREF = "enabled"; //NON-NLS |  | ||||||
|  |  | ||||||
|     private AudioNode splashSound; |  | ||||||
|     private AudioNode shipDestroyedSound; |  | ||||||
|     private AudioNode explosionSound; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if sound is enabled in the preferences. |  | ||||||
|      * |  | ||||||
|      * @return {@code true} if sound is enabled, {@code false} otherwise. |  | ||||||
|      */ |  | ||||||
|     public static boolean enabledInPreferences() { |  | ||||||
|         return PREFERENCES.getBoolean(ENABLED_PREF, true); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Toggles the game sound on or off. |  | ||||||
|      */ |  | ||||||
|     public void toggleSound() { |  | ||||||
|         setEnabled(!isEnabled()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Sets the enabled state of this AppState. |  | ||||||
|      * Overrides {@link com.jme3.app.state.AbstractAppState#setEnabled(boolean)} |  | ||||||
|      * |  | ||||||
|      * @param enabled {@code true} to enable the AppState, {@code false} to disable it. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void setEnabled(boolean enabled) { |  | ||||||
|         if (isEnabled() == enabled) return; |  | ||||||
|         super.setEnabled(enabled); |  | ||||||
|         LOGGER.log(Level.INFO, "Sound enabled: {0}", enabled); //NON-NLS |  | ||||||
|         PREFERENCES.putBoolean(ENABLED_PREF, enabled); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Initializes the sound effects for the game. |  | ||||||
|      * Overrides {@link AbstractAppState#initialize(AppStateManager, Application)} |  | ||||||
|      * |  | ||||||
|      * @param stateManager The state manager |  | ||||||
|      * @param app          The application |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void initialize(AppStateManager stateManager, Application app) { |  | ||||||
|         super.initialize(stateManager, app); |  | ||||||
|         shipDestroyedSound = loadSound(app, "Sound/Effects/sunken.wav"); //NON-NLS |  | ||||||
|         splashSound = loadSound(app, "Sound/Effects/splash.wav"); //NON-NLS |  | ||||||
|         explosionSound = loadSound(app, "Sound/Effects/explosion.wav"); //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Loads a sound from the specified file. |  | ||||||
|      * |  | ||||||
|      * @param app  The application |  | ||||||
|      * @param name The name of the sound file. |  | ||||||
|      * @return The loaded AudioNode. |  | ||||||
|      */ |  | ||||||
|     private AudioNode loadSound(Application app, String name) { |  | ||||||
|         try { |  | ||||||
|             final AudioNode sound = new AudioNode(app.getAssetManager(), name, AudioData.DataType.Buffer); |  | ||||||
|             sound.setLooping(false); |  | ||||||
|             sound.setPositional(false); |  | ||||||
|             return sound; |  | ||||||
|         } |  | ||||||
|         catch (AssetLoadException | AssetNotFoundException ex) { |  | ||||||
|             LOGGER.log(Level.ERROR, ex.getMessage(), ex); |  | ||||||
|         } |  | ||||||
|         return null; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Plays the splash sound effect. |  | ||||||
|      */ |  | ||||||
|     public void splash() { |  | ||||||
|         if (isEnabled() && splashSound != null) |  | ||||||
|             splashSound.playInstance(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Plays the explosion sound effect. |  | ||||||
|      */ |  | ||||||
|     public void explosion() { |  | ||||||
|         if (isEnabled() && explosionSound != null) |  | ||||||
|             explosionSound.playInstance(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Plays sound effect when a ship has been destroyed. |  | ||||||
|      */ |  | ||||||
|     public void shipDestroyed() { |  | ||||||
|         if (isEnabled() && shipDestroyedSound != null) |  | ||||||
|             shipDestroyedSound.playInstance(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public void receivedEvent(SoundEvent event) { |  | ||||||
|         switch (event.sound()) { |  | ||||||
|             case EXPLOSION -> explosion(); |  | ||||||
|             case SPLASH -> splash(); |  | ||||||
|             case DESTROYED_SHIP -> shipDestroyed(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,143 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.client; |  | ||||||
|  |  | ||||||
| import com.simsilica.lemur.Button; |  | ||||||
| import com.simsilica.lemur.Checkbox; |  | ||||||
| import com.simsilica.lemur.Label; |  | ||||||
| import com.simsilica.lemur.style.ElementId; |  | ||||||
| import pp.dialog.Dialog; |  | ||||||
| import pp.dialog.StateCheckboxModel; |  | ||||||
| import pp.dialog.TextInputDialog; |  | ||||||
|  |  | ||||||
| import java.io.File; |  | ||||||
| import java.io.IOException; |  | ||||||
| import java.util.prefs.Preferences; |  | ||||||
|  |  | ||||||
| import static pp.battleship.Resources.lookup; |  | ||||||
| import static pp.util.PreferencesUtils.getPreferences; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * The Menu class represents the main menu in the Battleship game application. |  | ||||||
|  * It extends the Dialog class and provides functionalities for loading, saving, |  | ||||||
|  * returning to the game, and quitting the application. |  | ||||||
|  */ |  | ||||||
| class Menu extends Dialog { |  | ||||||
|     private static final Preferences PREFERENCES = getPreferences(Menu.class); |  | ||||||
|     private static final String LAST_PATH = "last.file.path"; |  | ||||||
|     private final BattleshipApp app; |  | ||||||
|     private final Button loadButton = new Button(lookup("menu.map.load")); |  | ||||||
|     private final Button saveButton = new Button(lookup("menu.map.save")); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs the Menu dialog for the Battleship application. |  | ||||||
|      * |  | ||||||
|      * @param app the BattleshipApp instance |  | ||||||
|      */ |  | ||||||
|     public Menu(BattleshipApp app) { |  | ||||||
|         super(app.getDialogManager()); |  | ||||||
|         this.app = app; |  | ||||||
|         addChild(new Label(lookup("battleship.name"), new ElementId("header"))); //NON-NLS |  | ||||||
|         addChild(new Checkbox(lookup("menu.sound-enabled"), |  | ||||||
|                               new StateCheckboxModel(app, GameSound.class))); |  | ||||||
|         addChild(loadButton) |  | ||||||
|                 .addClickCommands(s -> ifTopDialog(this::loadDialog)); |  | ||||||
|         addChild(saveButton) |  | ||||||
|                 .addClickCommands(s -> ifTopDialog(this::saveDialog)); |  | ||||||
|         addChild(new Button(lookup("menu.return-to-game"))) |  | ||||||
|                 .addClickCommands(s -> ifTopDialog(this::close)); |  | ||||||
|         addChild(new Button(lookup("menu.quit"))) |  | ||||||
|                 .addClickCommands(s -> ifTopDialog(app::closeApp)); |  | ||||||
|         update(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Updates the state of the load and save buttons based on the game logic. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void update() { |  | ||||||
|         loadButton.setEnabled(app.getGameLogic().mayLoadMap()); |  | ||||||
|         saveButton.setEnabled(app.getGameLogic().maySaveMap()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * As an escape action, this method closes the menu if it is the top dialog. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void escape() { |  | ||||||
|         close(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Functional interface for file actions. |  | ||||||
|      */ |  | ||||||
|     @FunctionalInterface |  | ||||||
|     private interface FileAction { |  | ||||||
|         /** |  | ||||||
|          * Executes a file action. |  | ||||||
|          * |  | ||||||
|          * @param file the file to be processed |  | ||||||
|          * @throws IOException if an I/O error occurs |  | ||||||
|          */ |  | ||||||
|         void run(File file) throws IOException; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles the file action for the provided dialog. |  | ||||||
|      * |  | ||||||
|      * @param fileAction the file action to be executed |  | ||||||
|      * @param dialog     the dialog providing the file input |  | ||||||
|      */ |  | ||||||
|     private void handle(FileAction fileAction, TextInputDialog dialog) { |  | ||||||
|         try { |  | ||||||
|             final String path = dialog.getInput().getText(); |  | ||||||
|             PREFERENCES.put(LAST_PATH, path); |  | ||||||
|             fileAction.run(new File(path)); |  | ||||||
|             dialog.close(); |  | ||||||
|         } |  | ||||||
|         catch (IOException e) { |  | ||||||
|             app.errorDialog(e.getLocalizedMessage()); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Shows a file dialog for loading or saving files. |  | ||||||
|      * |  | ||||||
|      * @param fileAction the action to perform with the selected file |  | ||||||
|      * @param label      the label for the dialog |  | ||||||
|      */ |  | ||||||
|     private void fileDialog(FileAction fileAction, String label) { |  | ||||||
|         final TextInputDialog dialog = |  | ||||||
|                 TextInputDialog.builder(app.getDialogManager()) |  | ||||||
|                                .setLabel(lookup("label.file")) |  | ||||||
|                                .setFocus(TextInputDialog::getInput) |  | ||||||
|                                .setTitle(label) |  | ||||||
|                                .setOkButton(lookup("button.ok"), d -> handle(fileAction, d)) |  | ||||||
|                                .setNoButton(lookup("button.cancel")) |  | ||||||
|                                .setOkClose(false) |  | ||||||
|                                .build(); |  | ||||||
|         final String path = PREFERENCES.get(LAST_PATH, null); |  | ||||||
|         if (path != null) |  | ||||||
|             dialog.getInput().setText(path.trim()); |  | ||||||
|         dialog.open(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Shows the load dialog for loading maps. |  | ||||||
|      */ |  | ||||||
|     private void loadDialog() { |  | ||||||
|         fileDialog(app.getGameLogic()::loadMap, lookup("menu.map.load")); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Shows the save dialog for saving maps. |  | ||||||
|      */ |  | ||||||
|     private void saveDialog() { |  | ||||||
|         fileDialog(app.getGameLogic()::saveMap, lookup("menu.map.save")); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,153 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.client; |  | ||||||
|  |  | ||||||
| import com.simsilica.lemur.Container; |  | ||||||
| import com.simsilica.lemur.Label; |  | ||||||
| import com.simsilica.lemur.TextField; |  | ||||||
| import com.simsilica.lemur.component.SpringGridLayout; |  | ||||||
| import pp.dialog.Dialog; |  | ||||||
| import pp.dialog.DialogBuilder; |  | ||||||
| import pp.dialog.SimpleDialog; |  | ||||||
|  |  | ||||||
| import java.lang.System.Logger; |  | ||||||
| import java.lang.System.Logger.Level; |  | ||||||
| import java.util.concurrent.ExecutionException; |  | ||||||
| import java.util.concurrent.Future; |  | ||||||
|  |  | ||||||
| import static pp.battleship.Resources.lookup; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Represents a dialog for setting up a network connection in the Battleship game. |  | ||||||
|  * Allows users to specify the host and port for connecting to a game server. |  | ||||||
|  */ |  | ||||||
| class NetworkDialog extends SimpleDialog { |  | ||||||
|     private static final Logger LOGGER = System.getLogger(NetworkDialog.class.getName()); |  | ||||||
|     private static final String LOCALHOST = "localhost"; //NON-NLS |  | ||||||
|     private static final String DEFAULT_PORT = "1234"; //NON-NLS |  | ||||||
|     private final NetworkSupport network; |  | ||||||
|     private final TextField host = new TextField(LOCALHOST); |  | ||||||
|     private final TextField port = new TextField(DEFAULT_PORT); |  | ||||||
|     private String hostname; |  | ||||||
|     private int portNumber; |  | ||||||
|     private Future<Object> connectionFuture; |  | ||||||
|     private Dialog progressDialog; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a new NetworkDialog. |  | ||||||
|      * |  | ||||||
|      * @param network The NetworkSupport instance to be used for network operations. |  | ||||||
|      */ |  | ||||||
|     NetworkDialog(NetworkSupport network) { |  | ||||||
|         super(network.getApp().getDialogManager()); |  | ||||||
|         this.network = network; |  | ||||||
|         host.setSingleLine(true); |  | ||||||
|         host.setPreferredWidth(400f); |  | ||||||
|         port.setSingleLine(true); |  | ||||||
|  |  | ||||||
|         final BattleshipApp app = network.getApp(); |  | ||||||
|         final Container input = new Container(new SpringGridLayout()); |  | ||||||
|         input.addChild(new Label(lookup("host.name") + ":  ")); |  | ||||||
|         input.addChild(host, 1); |  | ||||||
|         input.addChild(new Label(lookup("port.number") + ":  ")); |  | ||||||
|         input.addChild(port, 1); |  | ||||||
|  |  | ||||||
|         DialogBuilder.simple(app.getDialogManager()) |  | ||||||
|                      .setTitle(lookup("server.dialog")) |  | ||||||
|                      .setExtension(d -> d.addChild(input)) |  | ||||||
|                      .setOkButton(lookup("button.connect"), d -> connect()) |  | ||||||
|                      .setNoButton(lookup("button.cancel"), app::closeApp) |  | ||||||
|                      .setOkClose(false) |  | ||||||
|                      .setNoClose(false) |  | ||||||
|                      .build(this); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles the action for the connect button in the connection dialog. |  | ||||||
|      * Tries to parse the port number and initiate connection to the server. |  | ||||||
|      */ |  | ||||||
|     private void connect() { |  | ||||||
|         LOGGER.log(Level.INFO, "connect to host={0}, port={1}", host, port); //NON-NLS |  | ||||||
|         try { |  | ||||||
|             hostname = host.getText().trim().isEmpty() ? LOCALHOST : host.getText(); |  | ||||||
|             portNumber = Integer.parseInt(port.getText()); |  | ||||||
|             openProgressDialog(); |  | ||||||
|             connectionFuture = network.getApp().getExecutor().submit(this::initNetwork); |  | ||||||
|         } |  | ||||||
|         catch (NumberFormatException e) { |  | ||||||
|             network.getApp().errorDialog(lookup("port.must.be.integer")); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates a dialog indicating that the connection is in progress. |  | ||||||
|      */ |  | ||||||
|     private void openProgressDialog() { |  | ||||||
|         progressDialog = DialogBuilder.simple(network.getApp().getDialogManager()) |  | ||||||
|                                       .setText(lookup("label.connecting")) |  | ||||||
|                                       .build(); |  | ||||||
|         progressDialog.open(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Tries to initialize the network connection. |  | ||||||
|      * |  | ||||||
|      * @throws RuntimeException If an error occurs when creating the client. |  | ||||||
|      */ |  | ||||||
|     private Object initNetwork() { |  | ||||||
|         try { |  | ||||||
|             network.initNetwork(hostname, portNumber); |  | ||||||
|             return null; |  | ||||||
|         } |  | ||||||
|         catch (Exception e) { |  | ||||||
|             throw new RuntimeException(e); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * This method is called by {@linkplain pp.dialog.DialogManager#update(float)} for periodically |  | ||||||
|      * updating this dialog. T |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void update(float delta) { |  | ||||||
|         if (connectionFuture != null && connectionFuture.isDone()) |  | ||||||
|             try { |  | ||||||
|                 connectionFuture.get(); |  | ||||||
|                 success(); |  | ||||||
|             } |  | ||||||
|             catch (ExecutionException e) { |  | ||||||
|                 failure(e.getCause()); |  | ||||||
|             } |  | ||||||
|             catch (InterruptedException e) { |  | ||||||
|                 LOGGER.log(Level.WARNING, "Interrupted!", e); //NON-NLS |  | ||||||
|                 Thread.currentThread().interrupt(); |  | ||||||
|             } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles a successful connection to the game server. |  | ||||||
|      */ |  | ||||||
|     private void success() { |  | ||||||
|         connectionFuture = null; |  | ||||||
|         progressDialog.close(); |  | ||||||
|         this.close(); |  | ||||||
|         network.getApp().setInfoText(lookup("wait.for.an.opponent")); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles a failed connection attempt. |  | ||||||
|      * |  | ||||||
|      * @param e The cause of the failure. |  | ||||||
|      */ |  | ||||||
|     private void failure(Throwable e) { |  | ||||||
|         connectionFuture = null; |  | ||||||
|         progressDialog.close(); |  | ||||||
|         network.getApp().errorDialog(lookup("server.connection.failed")); |  | ||||||
|         network.getApp().setInfoText(e.getLocalizedMessage()); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,152 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.client; |  | ||||||
|  |  | ||||||
| import com.jme3.network.Client; |  | ||||||
| import com.jme3.network.ClientStateListener; |  | ||||||
| import com.jme3.network.Message; |  | ||||||
| import com.jme3.network.MessageListener; |  | ||||||
| import com.jme3.network.Network; |  | ||||||
| import pp.battleship.game.client.ServerConnection; |  | ||||||
| import pp.battleship.message.client.ClientMessage; |  | ||||||
| import pp.battleship.message.server.ServerMessage; |  | ||||||
|  |  | ||||||
| import java.io.IOException; |  | ||||||
| import java.lang.System.Logger; |  | ||||||
| import java.lang.System.Logger.Level; |  | ||||||
|  |  | ||||||
| import static pp.battleship.Resources.lookup; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Manages the network connection for the Battleship application. |  | ||||||
|  * Handles connecting to and disconnecting from the server, and sending messages. |  | ||||||
|  */ |  | ||||||
| class NetworkSupport implements MessageListener<Client>, ClientStateListener, ServerConnection { |  | ||||||
|     private static final Logger LOGGER = System.getLogger(NetworkSupport.class.getName()); |  | ||||||
|     private final BattleshipApp app; |  | ||||||
|     private Client client; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a NetworkSupport instance for the given Battleship application. |  | ||||||
|      * |  | ||||||
|      * @param app The Battleship application instance. |  | ||||||
|      */ |  | ||||||
|     public NetworkSupport(BattleshipApp app) { |  | ||||||
|         this.app = app; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the Battleship application instance. |  | ||||||
|      * |  | ||||||
|      * @return Battleship application instance |  | ||||||
|      */ |  | ||||||
|     BattleshipApp getApp() { |  | ||||||
|         return app; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if there is a connection to the game server. |  | ||||||
|      * |  | ||||||
|      * @return true if there is a connection to the game server, false otherwise. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public boolean isConnected() { |  | ||||||
|         return client != null && client.isConnected(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Attempts to join the game if there is no connection yet. |  | ||||||
|      * Opens a dialog for the user to enter the host and port information. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void connect() { |  | ||||||
|         if (client == null) |  | ||||||
|             new NetworkDialog(this).open(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Closes the client connection. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void disconnect() { |  | ||||||
|         if (client == null) return; |  | ||||||
|         client.close(); |  | ||||||
|         client = null; |  | ||||||
|         LOGGER.log(Level.INFO, "client closed"); //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Initializes the network connection. |  | ||||||
|      * |  | ||||||
|      * @param host The server's address. |  | ||||||
|      * @param port The server's port. |  | ||||||
|      * @throws IOException If an I/O error occurs when creating the client. |  | ||||||
|      */ |  | ||||||
|     void initNetwork(String host, int port) throws IOException { |  | ||||||
|         if (client != null) |  | ||||||
|             throw new IllegalStateException("trying to join a game again"); |  | ||||||
|         client = Network.connectToServer(host, port); |  | ||||||
|         client.start(); |  | ||||||
|         client.addMessageListener(this); |  | ||||||
|         client.addClientStateListener(this); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Called when a message is received from the server. |  | ||||||
|      * |  | ||||||
|      * @param client  The client instance that received the message. |  | ||||||
|      * @param message The message received from the server. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void messageReceived(Client client, Message message) { |  | ||||||
|         LOGGER.log(Level.INFO, "message received from server: {0}", message); //NON-NLS |  | ||||||
|         if (message instanceof ServerMessage serverMessage) |  | ||||||
|             app.enqueue(() -> serverMessage.accept(app.getGameLogic())); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Called when the client has successfully connected to the server. |  | ||||||
|      * |  | ||||||
|      * @param client The client that connected to the server. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void clientConnected(Client client) { |  | ||||||
|         LOGGER.log(Level.INFO, "Client connected: {0}", client); //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Called when the client is disconnected from the server. |  | ||||||
|      * |  | ||||||
|      * @param client         The client that was disconnected. |  | ||||||
|      * @param disconnectInfo Information about the disconnection. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void clientDisconnected(Client client, DisconnectInfo disconnectInfo) { |  | ||||||
|         LOGGER.log(Level.INFO, "Client {0} disconnected: {1}", client, disconnectInfo); //NON-NLS |  | ||||||
|         if (this.client != client) |  | ||||||
|             throw new IllegalArgumentException("parameter value must be client"); |  | ||||||
|         LOGGER.log(Level.INFO, "client still connected: {0}", client.isConnected()); //NON-NLS |  | ||||||
|         this.client = null; |  | ||||||
|         disconnect(); |  | ||||||
|         app.enqueue(() -> app.setInfoText(lookup("lost.connection.to.server"))); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Sends the specified message to the server. |  | ||||||
|      * |  | ||||||
|      * @param message The message to be sent to the server. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void send(ClientMessage message) { |  | ||||||
|         LOGGER.log(Level.INFO, "sending {0}", message); //NON-NLS |  | ||||||
|         if (client == null) |  | ||||||
|             app.errorDialog(lookup("lost.connection.to.server")); |  | ||||||
|         else |  | ||||||
|             client.send(message); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,122 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.client.gui; |  | ||||||
|  |  | ||||||
| import com.jme3.input.controls.ActionListener; |  | ||||||
| import com.jme3.scene.Node; |  | ||||||
| import com.jme3.system.AppSettings; |  | ||||||
| import pp.battleship.client.BattleshipAppState; |  | ||||||
| import pp.battleship.model.IntPoint; |  | ||||||
|  |  | ||||||
| import java.lang.System.Logger; |  | ||||||
| import java.lang.System.Logger.Level; |  | ||||||
|  |  | ||||||
| import static pp.battleship.client.BattleshipApp.CLICK; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Represents the state responsible for managing the battle interface within the Battleship game. |  | ||||||
|  * This state handles the display and interaction of the battle map, including the opponent's map. |  | ||||||
|  * It manages GUI components, input events, and the layout of the interface when this state is enabled. |  | ||||||
|  */ |  | ||||||
| public class BattleAppState extends BattleshipAppState { |  | ||||||
|     private static final Logger LOGGER = System.getLogger(BattleAppState.class.getName()); |  | ||||||
|     private static final float DEPTH = 0f; |  | ||||||
|     private static final float GAP = 20f; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * A listener for handling click events in the battle interface. |  | ||||||
|      * When a click is detected, it triggers the corresponding actions on the opponent's map. |  | ||||||
|      */ |  | ||||||
|     private final ActionListener clickListener = (name, isPressed, tpf) -> click(isPressed); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The root node for all GUI components in the battle state. |  | ||||||
|      */ |  | ||||||
|     private final Node battleNode = new Node("Battle"); //NON-NLS |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * A view representing the opponent's map in the GUI. |  | ||||||
|      */ |  | ||||||
|     private MapView opponentMapView; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Enables the battle state by initializing, laying out, and adding GUI components. |  | ||||||
|      * Attaches the components to the GUI node and registers input listeners. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     protected void enableState() { |  | ||||||
|         battleNode.detachAllChildren(); |  | ||||||
|         initializeGuiComponents(); |  | ||||||
|         layoutGuiComponents(); |  | ||||||
|         addGuiComponents(); |  | ||||||
|         getApp().getGuiNode().attachChild(battleNode); |  | ||||||
|         getApp().getInputManager().addListener(clickListener, CLICK); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Disables the battle state by removing GUI components and unregistering input listeners. |  | ||||||
|      * Also handles cleanup of resources, such as the opponent's map view. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     protected void disableState() { |  | ||||||
|         getApp().getGuiNode().detachChild(battleNode); |  | ||||||
|         getApp().getInputManager().removeListener(clickListener); |  | ||||||
|         if (opponentMapView != null) { |  | ||||||
|             opponentMapView.unregister(); |  | ||||||
|             opponentMapView = null; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Initializes the GUI components used in the battle state. |  | ||||||
|      * Creates the opponent's map view and adds a grid overlay to it. |  | ||||||
|      */ |  | ||||||
|     private void initializeGuiComponents() { |  | ||||||
|         opponentMapView = new MapView(getGameLogic().getOpponentMap(), getApp()); |  | ||||||
|         opponentMapView.addGrid(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Adds the initialized GUI components to the battle node. |  | ||||||
|      * Currently, it attaches the opponent's map view to the node. |  | ||||||
|      */ |  | ||||||
|     private void addGuiComponents() { |  | ||||||
|         battleNode.attachChild(opponentMapView.getNode()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Lays out the GUI components within the window, positioning them appropriately. |  | ||||||
|      * The opponent's map view is positioned based on the window's dimensions and a specified gap. |  | ||||||
|      */ |  | ||||||
|     private void layoutGuiComponents() { |  | ||||||
|         final AppSettings s = getApp().getContext().getSettings(); |  | ||||||
|         final float mapWidth = opponentMapView.getWidth(); |  | ||||||
|         final float mapHeight = opponentMapView.getHeight(); |  | ||||||
|         final float windowWidth = s.getWidth(); |  | ||||||
|         final float windowHeight = s.getHeight(); |  | ||||||
|  |  | ||||||
|         opponentMapView.getNode().setLocalTranslation(windowWidth - mapWidth - GAP, |  | ||||||
|                                                       windowHeight - mapHeight - GAP, |  | ||||||
|                                                       DEPTH); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles click events in the battle interface. If the event indicates a click (not a release), |  | ||||||
|      * it translates the cursor position to the model's coordinate system and triggers the game logic |  | ||||||
|      * for interacting with the opponent's map. |  | ||||||
|      * |  | ||||||
|      * @param isPressed whether the mouse button is currently pressed (true) or released (false) |  | ||||||
|      */ |  | ||||||
|     private void click(boolean isPressed) { |  | ||||||
|         if (!isPressed || showsDialog()) |  | ||||||
|             return; |  | ||||||
|         final IntPoint cursorPos = opponentMapView.mouseToModel(getApp().getInputManager().getCursorPosition()); |  | ||||||
|         LOGGER.log(Level.DEBUG, "click: {0}", cursorPos); //NON-NLS |  | ||||||
|         getGameLogic().clickOpponentMap(cursorPos); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,173 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.client.gui; |  | ||||||
|  |  | ||||||
| import com.jme3.input.controls.ActionListener; |  | ||||||
| import com.jme3.math.Vector2f; |  | ||||||
| import com.jme3.scene.Node; |  | ||||||
| import com.jme3.system.AppSettings; |  | ||||||
| import com.simsilica.lemur.Button; |  | ||||||
| import com.simsilica.lemur.Container; |  | ||||||
| import pp.battleship.client.BattleshipAppState; |  | ||||||
|  |  | ||||||
| import java.lang.System.Logger; |  | ||||||
| import java.lang.System.Logger.Level; |  | ||||||
|  |  | ||||||
| import static pp.battleship.Resources.lookup; |  | ||||||
| import static pp.battleship.client.BattleshipApp.CLICK; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * EditorState manages the editor mode in the Battleship game, |  | ||||||
|  * allowing players to place and rotate ships. |  | ||||||
|  */ |  | ||||||
| public class EditorAppState extends BattleshipAppState { |  | ||||||
|     private static final Logger LOGGER = System.getLogger(EditorAppState.class.getName()); |  | ||||||
|     private static final float DEPTH = 0f; |  | ||||||
|     private static final float GAP = 20f; |  | ||||||
|  |  | ||||||
|     private final ActionListener clickListener = (name, isPressed, tpf) -> click(isPressed); |  | ||||||
|     private final Node editorNode = new Node("Editor"); //NON-NLS |  | ||||||
|     private Container buttonContainer; |  | ||||||
|     private Button rotateButton; |  | ||||||
|     private Button readyButton; |  | ||||||
|     private MapView ownMapView; |  | ||||||
|     private MapView harborView; |  | ||||||
|     private Vector2f oldCursorPosition; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Enables the editor state by attaching necessary nodes and listeners. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     protected void enableState() { |  | ||||||
|         editorNode.detachAllChildren(); |  | ||||||
|         initializeGuiComponents(); |  | ||||||
|         addGuiComponents(); |  | ||||||
|         layoutGuiComponents(); |  | ||||||
|         getApp().getGuiNode().attachChild(editorNode); |  | ||||||
|         getApp().getInputManager().addListener(clickListener, CLICK); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Disables the editor state by detaching nodes and removing listeners. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     protected void disableState() { |  | ||||||
|         getApp().getGuiNode().detachChild(editorNode); |  | ||||||
|         getApp().getInputManager().removeListener(clickListener); |  | ||||||
|         if (ownMapView != null) { |  | ||||||
|             ownMapView.unregister(); |  | ||||||
|             ownMapView = null; |  | ||||||
|         } |  | ||||||
|         if (harborView != null) { |  | ||||||
|             harborView.unregister(); |  | ||||||
|             harborView = null; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Updates the editor state, handling cursor movement and enabling buttons. |  | ||||||
|      * |  | ||||||
|      * @param tpf Time per frame |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void update(float tpf) { |  | ||||||
|         super.update(tpf); |  | ||||||
|         cursorMovement(); |  | ||||||
|         enableButtons(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Enables or disables buttons based on the logic state. |  | ||||||
|      */ |  | ||||||
|     private void enableButtons() { |  | ||||||
|         readyButton.setEnabled(getGameLogic().isMapComplete()); |  | ||||||
|         rotateButton.setEnabled(getGameLogic().movingShip()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles cursor movement for previewing ship placement. |  | ||||||
|      */ |  | ||||||
|     private void cursorMovement() { |  | ||||||
|         if (!getGameLogic().movingShip() || ownMapView == null || showsDialog()) |  | ||||||
|             return; |  | ||||||
|         final Vector2f cursorPosition = getApp().getInputManager().getCursorPosition(); |  | ||||||
|         if (!cursorPosition.equals(oldCursorPosition)) { |  | ||||||
|             oldCursorPosition = new Vector2f(cursorPosition); |  | ||||||
|             getGameLogic().movePreview(ownMapView.mouseToModel(cursorPosition)); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Initializes the GUI components for the editor. |  | ||||||
|      */ |  | ||||||
|     private void initializeGuiComponents() { |  | ||||||
|         ownMapView = new MapView(getGameLogic().getOwnMap(), getApp()); |  | ||||||
|         harborView = new MapView(getGameLogic().getHarbor(), getApp()); |  | ||||||
|         ownMapView.addGrid(); |  | ||||||
|         rotateButton = new Button(lookup("button.rotate")); |  | ||||||
|         readyButton = new Button(lookup("button.ready")); |  | ||||||
|         rotateButton.addClickCommands(e -> { |  | ||||||
|             if (!showsDialog()) |  | ||||||
|                 getGameLogic().rotateShip(); |  | ||||||
|         }); |  | ||||||
|         readyButton.addClickCommands(e -> { |  | ||||||
|             if (!showsDialog()) |  | ||||||
|                 getGameLogic().mapFinished(); |  | ||||||
|         }); |  | ||||||
|         buttonContainer = new Container(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Adds the GUI components to the editor node. |  | ||||||
|      */ |  | ||||||
|     private void addGuiComponents() { |  | ||||||
|         buttonContainer.addChild(rotateButton, 0, 0); |  | ||||||
|         buttonContainer.addChild(readyButton, 0, 1); |  | ||||||
|         editorNode.attachChild(ownMapView.getNode()); |  | ||||||
|         editorNode.attachChild(harborView.getNode()); |  | ||||||
|         editorNode.attachChild(buttonContainer); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Lays out the GUI components on the screen. |  | ||||||
|      */ |  | ||||||
|     private void layoutGuiComponents() { |  | ||||||
|         final AppSettings s = getApp().getContext().getSettings(); |  | ||||||
|         final float harborWidth = harborView.getWidth(); |  | ||||||
|         final float harborHeight = harborView.getHeight(); |  | ||||||
|         final float ownMapWidth = ownMapView.getWidth(); |  | ||||||
|         final float ownMapHeight = ownMapView.getHeight(); |  | ||||||
|         final float windowWidth = s.getWidth(); |  | ||||||
|         final float windowHeight = s.getHeight(); |  | ||||||
|  |  | ||||||
|         ownMapView.getNode() |  | ||||||
|                   .setLocalTranslation(0.5f * (windowWidth - harborWidth - ownMapWidth - GAP), |  | ||||||
|                                        0.5f * (windowHeight - ownMapHeight), |  | ||||||
|                                        DEPTH); |  | ||||||
|  |  | ||||||
|         harborView.getNode() |  | ||||||
|                   .setLocalTranslation(0.5f * (windowWidth - harborWidth + ownMapWidth + GAP), |  | ||||||
|                                        0.5f * (windowHeight - harborHeight), |  | ||||||
|                                        DEPTH); |  | ||||||
|  |  | ||||||
|         buttonContainer.setLocalTranslation(0.5f * (windowWidth - harborWidth - ownMapWidth - GAP), |  | ||||||
|                                             0.5f * (windowHeight - ownMapHeight - GAP), |  | ||||||
|                                             DEPTH); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles click events to place or rotate ships on the maps. |  | ||||||
|      */ |  | ||||||
|     private void click(boolean isPressed) { |  | ||||||
|         if (!isPressed || showsDialog()) return; |  | ||||||
|         final Vector2f cursorPos = getApp().getInputManager().getCursorPosition(); |  | ||||||
|         LOGGER.log(Level.DEBUG, "click: {0}", cursorPos); //NON-NLS |  | ||||||
|         getGameLogic().clickHarbor(harborView.mouseToModel(cursorPos)); |  | ||||||
|         getGameLogic().clickOwnMap(ownMapView.mouseToModel(cursorPos)); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,191 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.client.gui; |  | ||||||
|  |  | ||||||
| import com.jme3.material.Material; |  | ||||||
| import com.jme3.material.RenderState.BlendMode; |  | ||||||
| import com.jme3.math.ColorRGBA; |  | ||||||
| import com.jme3.math.Vector2f; |  | ||||||
| import com.jme3.math.Vector3f; |  | ||||||
| import com.jme3.scene.Geometry; |  | ||||||
| import com.jme3.scene.Node; |  | ||||||
| import com.jme3.scene.Spatial.CullHint; |  | ||||||
| import com.jme3.scene.shape.Quad; |  | ||||||
| import pp.battleship.client.BattleshipApp; |  | ||||||
| import pp.battleship.model.IntPoint; |  | ||||||
| import pp.battleship.model.ShipMap; |  | ||||||
| import pp.util.FloatPoint; |  | ||||||
| import pp.util.Position; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Represents the visual view of a {@link ShipMap}, used to display the map structure such as the player's map, harbor, |  | ||||||
|  * and opponent's map. This class handles the graphical representation of the map, including background setup, grid lines, |  | ||||||
|  * and interaction between the model and the view. |  | ||||||
|  */ |  | ||||||
| class MapView { |  | ||||||
|     private static final float FIELD_SIZE = 40f; |  | ||||||
|     private static final float GRID_LINE_WIDTH = 2f; |  | ||||||
|     private static final float BACKGROUND_DEPTH = -4f; |  | ||||||
|     private static final float GRID_DEPTH = -1f; |  | ||||||
|     private static final ColorRGBA BACKGROUND_COLOR = new ColorRGBA(0, 0.05f, 0.05f, 0.5f); |  | ||||||
|     private static final ColorRGBA GRID_COLOR = ColorRGBA.Green; |  | ||||||
|  |  | ||||||
|     // Reference to the main application and the ship map being visualized |  | ||||||
|     private final BattleshipApp app; |  | ||||||
|     private final Node mapNode = new Node("map"); // NON-NLS |  | ||||||
|     private final ShipMap map; |  | ||||||
|     private final MapViewSynchronizer synchronizer; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a new MapView for a given {@link ShipMap} and {@link BattleshipApp}. |  | ||||||
|      * Initializes the view by setting up the background and registering a synchronizer to listen to changes in the map. |  | ||||||
|      * |  | ||||||
|      * @param map the ship map to visualize |  | ||||||
|      * @param app the main application instance |  | ||||||
|      */ |  | ||||||
|     MapView(ShipMap map, BattleshipApp app) { |  | ||||||
|         this.map = map; |  | ||||||
|         this.app = app; |  | ||||||
|         this.synchronizer = new MapViewSynchronizer(this); |  | ||||||
|         setupBackground(); |  | ||||||
|         app.getGameLogic().addListener(synchronizer); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Unregisters the {@link MapViewSynchronizer} from the listener list of the ClientGameLogic, |  | ||||||
|      * stopping the view from receiving updates when the underlying {@link ShipMap} changes. |  | ||||||
|      * After calling this method, this MapView instance should no longer be used. |  | ||||||
|      */ |  | ||||||
|     void unregister() { |  | ||||||
|         app.getGameLogic().removeListener(synchronizer); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Gets the {@link ShipMap} associated with this view. |  | ||||||
|      * |  | ||||||
|      * @return the ship map |  | ||||||
|      */ |  | ||||||
|     public ShipMap getMap() { |  | ||||||
|         return map; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Gets the {@link BattleshipApp} instance associated with this view. |  | ||||||
|      * |  | ||||||
|      * @return the main application instance |  | ||||||
|      */ |  | ||||||
|     public BattleshipApp getApp() { |  | ||||||
|         return app; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Sets up the background of the map view using a quad geometry. |  | ||||||
|      * The background is configured with a semi-transparent color and placed at a specific depth. |  | ||||||
|      */ |  | ||||||
|     private void setupBackground() { |  | ||||||
|         final Material mat = new Material(app.getAssetManager(), "Common/MatDefs/Misc/Unshaded.j3md"); // NON-NLS |  | ||||||
|         mat.setColor("Color", BACKGROUND_COLOR); // NON-NLS |  | ||||||
|         mat.getAdditionalRenderState().setBlendMode(BlendMode.Alpha); |  | ||||||
|         final Position corner = modelToView(map.getWidth(), map.getHeight()); |  | ||||||
|         final Geometry background = new Geometry("MapBackground", new Quad(corner.getX(), corner.getY())); |  | ||||||
|         background.setMaterial(mat); |  | ||||||
|         background.setLocalTranslation(0f, 0f, BACKGROUND_DEPTH); |  | ||||||
|         background.setCullHint(CullHint.Never); |  | ||||||
|         mapNode.attachChild(background); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Adds grid lines to the map view to visually separate the fields within the map. |  | ||||||
|      * The grid lines are drawn based on the dimensions of the ship map. |  | ||||||
|      */ |  | ||||||
|     public void addGrid() { |  | ||||||
|         for (int x = 0; x <= map.getWidth(); x++) { |  | ||||||
|             final Position f = modelToView(x, 0); |  | ||||||
|             final Position t = modelToView(x, map.getHeight()); |  | ||||||
|             mapNode.attachChild(gridLine(f, t)); |  | ||||||
|         } |  | ||||||
|         for (int y = 0; y <= map.getHeight(); y++) { |  | ||||||
|             final Position f = modelToView(0, y); |  | ||||||
|             final Position t = modelToView(map.getWidth(), y); |  | ||||||
|             mapNode.attachChild(gridLine(f, t)); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Gets the root node containing all visual elements in this map view. |  | ||||||
|      * |  | ||||||
|      * @return the root node for the map view |  | ||||||
|      */ |  | ||||||
|     public Node getNode() { |  | ||||||
|         return mapNode; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Gets the total width of the map in view coordinates. |  | ||||||
|      * |  | ||||||
|      * @return the width of the map in view coordinates |  | ||||||
|      */ |  | ||||||
|     public float getWidth() { |  | ||||||
|         return FIELD_SIZE * map.getWidth(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Gets the total height of the map in view coordinates. |  | ||||||
|      * |  | ||||||
|      * @return the height of the map in view coordinates |  | ||||||
|      */ |  | ||||||
|     public float getHeight() { |  | ||||||
|         return FIELD_SIZE * map.getHeight(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Converts coordinates from view coordinates to model coordinates. |  | ||||||
|      * |  | ||||||
|      * @param x the x-coordinate in view space |  | ||||||
|      * @param y the y-coordinate in view space |  | ||||||
|      * @return the corresponding model coordinates as an {@link IntPoint} |  | ||||||
|      */ |  | ||||||
|     public IntPoint viewToModel(float x, float y) { |  | ||||||
|         return new IntPoint((int) Math.floor(x / FIELD_SIZE), (int) Math.floor(y / FIELD_SIZE)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Converts coordinates from model coordinates to view coordinates. |  | ||||||
|      * |  | ||||||
|      * @param x the x-coordinate in model space |  | ||||||
|      * @param y the y-coordinate in model space |  | ||||||
|      * @return the corresponding view coordinates as a {@link Position} |  | ||||||
|      */ |  | ||||||
|     public Position modelToView(float x, float y) { |  | ||||||
|         return new FloatPoint(x * FIELD_SIZE, y * FIELD_SIZE); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Converts the mouse position to model coordinates. |  | ||||||
|      * This method takes into account the map's transformation in the 3D scene. |  | ||||||
|      * |  | ||||||
|      * @param pos the 2D vector representing the mouse position in the view |  | ||||||
|      * @return the corresponding model coordinates as an {@link IntPoint} |  | ||||||
|      */ |  | ||||||
|     public IntPoint mouseToModel(Vector2f pos) { |  | ||||||
|         final Vector3f world = new Vector3f(pos.getX(), pos.getY(), 0f); |  | ||||||
|         final Vector3f view = mapNode.getWorldTransform().transformInverseVector(world, null); |  | ||||||
|         return viewToModel(view.getX(), view.getY()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates a visual representation of a grid line between two positions. |  | ||||||
|      * |  | ||||||
|      * @param p1 the start position of the grid line |  | ||||||
|      * @param p2 the end position of the grid line |  | ||||||
|      * @return a {@link Geometry} representing the grid line |  | ||||||
|      */ |  | ||||||
|     private Geometry gridLine(Position p1, Position p2) { |  | ||||||
|         return app.getDraw().makeFatLine(p1, p2, GRID_DEPTH, GRID_COLOR, GRID_LINE_WIDTH); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,125 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.client.gui; |  | ||||||
|  |  | ||||||
| import com.jme3.math.ColorRGBA; |  | ||||||
| import com.jme3.scene.Geometry; |  | ||||||
| import com.jme3.scene.Node; |  | ||||||
| import com.jme3.scene.Spatial; |  | ||||||
| import pp.battleship.model.Battleship; |  | ||||||
| import pp.battleship.model.Shot; |  | ||||||
| import pp.util.Position; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Synchronizes the visual representation of the ship map with the game model. |  | ||||||
|  * It handles the rendering of ships and shots on the map view, updating the view |  | ||||||
|  * whenever changes occur in the model. |  | ||||||
|  */ |  | ||||||
| class MapViewSynchronizer extends ShipMapSynchronizer { |  | ||||||
|     // Constants for rendering properties |  | ||||||
|     private static final float SHIP_LINE_WIDTH = 6f; |  | ||||||
|     private static final float SHOT_DEPTH = -2f; |  | ||||||
|     private static final float SHIP_DEPTH = 0f; |  | ||||||
|     private static final float INDENT = 4f; |  | ||||||
|  |  | ||||||
|     // Colors used for different visual elements |  | ||||||
|     private static final ColorRGBA HIT_COLOR = ColorRGBA.Red; |  | ||||||
|     private static final ColorRGBA MISS_COLOR = ColorRGBA.Blue; |  | ||||||
|     private static final ColorRGBA SHIP_BORDER_COLOR = ColorRGBA.White; |  | ||||||
|     private static final ColorRGBA PREVIEW_COLOR = ColorRGBA.Gray; |  | ||||||
|     private static final ColorRGBA ERROR_COLOR = ColorRGBA.Red; |  | ||||||
|  |  | ||||||
|     // The MapView associated with this synchronizer |  | ||||||
|     private final MapView view; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a new MapViewSynchronizer for the given MapView. |  | ||||||
|      * Initializes the synchronizer and adds existing elements from the model to the view. |  | ||||||
|      * |  | ||||||
|      * @param view the MapView to synchronize with the game model |  | ||||||
|      */ |  | ||||||
|     public MapViewSynchronizer(MapView view) { |  | ||||||
|         super(view.getMap(), view.getNode()); |  | ||||||
|         this.view = view; |  | ||||||
|         addExisting(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates a visual representation of a shot on the map. |  | ||||||
|      * A hit shot is represented in red, while a miss is represented in blue. |  | ||||||
|      * |  | ||||||
|      * @param shot the Shot object representing the shot in the model |  | ||||||
|      * @return a Spatial representing the shot on the map |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public Spatial visit(Shot shot) { |  | ||||||
|         // Convert the shot's model coordinates to view coordinates |  | ||||||
|         final Position p1 = view.modelToView(shot.getX(), shot.getY()); |  | ||||||
|         final Position p2 = view.modelToView(shot.getX() + 1, shot.getY() + 1); |  | ||||||
|         final ColorRGBA color = shot.isHit() ? HIT_COLOR : MISS_COLOR; |  | ||||||
|  |  | ||||||
|         // Create and return a rectangle representing the shot |  | ||||||
|         return view.getApp().getDraw().makeRectangle(p1.getX(), p1.getY(), |  | ||||||
|                                                      SHOT_DEPTH, |  | ||||||
|                                                      p2.getX() - p1.getX(), p2.getY() - p1.getY(), |  | ||||||
|                                                      color); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates a visual representation of a battleship on the map. |  | ||||||
|      * The ship's border color depends on its status: normal, valid preview, or invalid preview. |  | ||||||
|      * |  | ||||||
|      * @param ship the Battleship object representing the ship in the model |  | ||||||
|      * @return a Spatial representing the ship on the map |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public Spatial visit(Battleship ship) { |  | ||||||
|         // Create a node to represent the ship |  | ||||||
|         final Node shipNode = new Node("ship"); //NON-NLS |  | ||||||
|  |  | ||||||
|         // Convert the ship's model coordinates to view coordinates |  | ||||||
|         final Position p1 = view.modelToView(ship.getMinX(), ship.getMinY()); |  | ||||||
|         final Position p2 = view.modelToView(ship.getMaxX() + 1, ship.getMaxY() + 1); |  | ||||||
|  |  | ||||||
|         // Calculate the coordinates for the ship's bounding box |  | ||||||
|         final float x1 = p1.getX() + INDENT; |  | ||||||
|         final float y1 = p1.getY() + INDENT; |  | ||||||
|         final float x2 = p2.getX() - INDENT; |  | ||||||
|         final float y2 = p2.getY() - INDENT; |  | ||||||
|  |  | ||||||
|         // Determine the color based on the ship's status |  | ||||||
|         final ColorRGBA color = switch (ship.getStatus()) { |  | ||||||
|             case NORMAL -> SHIP_BORDER_COLOR; |  | ||||||
|             case VALID_PREVIEW -> PREVIEW_COLOR; |  | ||||||
|             case INVALID_PREVIEW -> ERROR_COLOR; |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         // Add the ship's borders to the node |  | ||||||
|         shipNode.attachChild(shipLine(x1, y1, x2, y1, color)); |  | ||||||
|         shipNode.attachChild(shipLine(x1, y2, x2, y2, color)); |  | ||||||
|         shipNode.attachChild(shipLine(x1, y1, x1, y2, color)); |  | ||||||
|         shipNode.attachChild(shipLine(x2, y1, x2, y2, color)); |  | ||||||
|  |  | ||||||
|         // Return the complete ship representation |  | ||||||
|         return shipNode; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates a line geometry representing part of the ship's border. |  | ||||||
|      * |  | ||||||
|      * @param x1    the starting x-coordinate of the line |  | ||||||
|      * @param y1    the starting y-coordinate of the line |  | ||||||
|      * @param x2    the ending x-coordinate of the line |  | ||||||
|      * @param y2    the ending y-coordinate of the line |  | ||||||
|      * @param color the color of the line |  | ||||||
|      * @return a Geometry representing the line |  | ||||||
|      */ |  | ||||||
|     private Geometry shipLine(float x1, float y1, float x2, float y2, ColorRGBA color) { |  | ||||||
|         return view.getApp().getDraw().makeFatLine(x1, y1, x2, y2, SHIP_DEPTH, color, SHIP_LINE_WIDTH); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,203 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.client.gui; |  | ||||||
|  |  | ||||||
| import com.jme3.app.Application; |  | ||||||
| import com.jme3.app.state.AppStateManager; |  | ||||||
| import com.jme3.asset.AssetManager; |  | ||||||
| import com.jme3.light.AmbientLight; |  | ||||||
| import com.jme3.light.DirectionalLight; |  | ||||||
| import com.jme3.material.Material; |  | ||||||
| import com.jme3.math.ColorRGBA; |  | ||||||
| import com.jme3.math.Vector2f; |  | ||||||
| import com.jme3.math.Vector3f; |  | ||||||
| import com.jme3.renderer.Camera; |  | ||||||
| import com.jme3.renderer.queue.RenderQueue.ShadowMode; |  | ||||||
| import com.jme3.scene.Geometry; |  | ||||||
| import com.jme3.scene.Node; |  | ||||||
| import com.jme3.scene.Spatial; |  | ||||||
| import com.jme3.scene.shape.Box; |  | ||||||
| import com.jme3.shadow.DirectionalLightShadowRenderer; |  | ||||||
| import com.jme3.shadow.EdgeFilteringMode; |  | ||||||
| import com.jme3.texture.Texture; |  | ||||||
| import com.jme3.util.SkyFactory; |  | ||||||
| import com.jme3.util.TangentBinormalGenerator; |  | ||||||
| import pp.battleship.client.BattleshipAppState; |  | ||||||
| import pp.battleship.model.ShipMap; |  | ||||||
| import pp.util.FloatMath; |  | ||||||
|  |  | ||||||
| import static pp.util.FloatMath.TWO_PI; |  | ||||||
| import static pp.util.FloatMath.cos; |  | ||||||
| import static pp.util.FloatMath.sin; |  | ||||||
| import static pp.util.FloatMath.sqrt; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Manages the rendering and visual aspects of the sea and sky in the Battleship game. |  | ||||||
|  * This state is responsible for setting up and updating the sea, sky, and lighting |  | ||||||
|  * conditions, and controls the camera to create a dynamic view of the game environment. |  | ||||||
|  */ |  | ||||||
| public class SeaAppState extends BattleshipAppState { |  | ||||||
|     /** |  | ||||||
|      * The path to the sea texture material. |  | ||||||
|      */ |  | ||||||
|     private static final String SEA_TEXTURE = "Textures/Terrain/Water/Water.j3m"; //NON-NLS |  | ||||||
|     private static final float ABOVE_SEA_LEVEL = 4f; |  | ||||||
|     private static final float INCLINATION = 2.5f; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The root node for all visual elements in this state. |  | ||||||
|      */ |  | ||||||
|     private final Node viewNode = new Node("view"); //NON-NLS |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The node containing the scene elements, such as the sea surface. |  | ||||||
|      */ |  | ||||||
|     private final Node sceneNode = new Node("scene"); //NON-NLS |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Synchronizes the sea's visual representation with the game logic. |  | ||||||
|      */ |  | ||||||
|     private SeaSynchronizer synchronizer; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The current angle of the camera around the center of the map. |  | ||||||
|      */ |  | ||||||
|     private float cameraAngle; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Initializes the state by setting up the sky, lights, and other visual components. |  | ||||||
|      * This method is called when the state is first attached to the state manager. |  | ||||||
|      * |  | ||||||
|      * @param stateManager the state manager |  | ||||||
|      * @param application  the application |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void initialize(AppStateManager stateManager, Application application) { |  | ||||||
|         super.initialize(stateManager, application); |  | ||||||
|         viewNode.attachChild(sceneNode); |  | ||||||
|         setupLights(); |  | ||||||
|         setupSky(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Enables the sea and sky state, setting up the scene and registering any necessary listeners. |  | ||||||
|      * This method is called when the state is set to active. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     protected void enableState() { |  | ||||||
|         sceneNode.detachAllChildren(); |  | ||||||
|         setupScene(); |  | ||||||
|         if (synchronizer == null) { |  | ||||||
|             synchronizer = new SeaSynchronizer(getApp(), sceneNode, getGameLogic().getOwnMap()); |  | ||||||
|             getGameLogic().addListener(synchronizer); |  | ||||||
|         } |  | ||||||
|         getApp().getRootNode().attachChild(viewNode); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Disables the sea and sky state, removing visual elements from the scene and unregistering listeners. |  | ||||||
|      * This method is called when the state is set to inactive. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     protected void disableState() { |  | ||||||
|         getApp().getRootNode().detachChild(viewNode); |  | ||||||
|         if (synchronizer != null) { |  | ||||||
|             getGameLogic().removeListener(synchronizer); |  | ||||||
|             synchronizer = null; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Updates the state each frame, moving the camera to simulate it circling around the map. |  | ||||||
|      * |  | ||||||
|      * @param tpf the time per frame (seconds) |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void update(float tpf) { |  | ||||||
|         super.update(tpf); |  | ||||||
|         cameraAngle += TWO_PI * 0.05f * tpf; |  | ||||||
|         adjustCamera(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Sets up the lighting for the scene, including directional and ambient lights. |  | ||||||
|      * Also configures shadows to enhance the visual depth of the scene. |  | ||||||
|      */ |  | ||||||
|     private void setupLights() { |  | ||||||
|         final AssetManager assetManager = getApp().getAssetManager(); |  | ||||||
|         final DirectionalLightShadowRenderer shRend = new DirectionalLightShadowRenderer(assetManager, 2048, 3); |  | ||||||
|         shRend.setLambda(0.55f); |  | ||||||
|         shRend.setShadowIntensity(0.6f); |  | ||||||
|         shRend.setEdgeFilteringMode(EdgeFilteringMode.Bilinear); |  | ||||||
|         getApp().getViewPort().addProcessor(shRend); |  | ||||||
|  |  | ||||||
|         final DirectionalLight sun = new DirectionalLight(); |  | ||||||
|         sun.setDirection(new Vector3f(-1f, -0.7f, -1f).normalizeLocal()); |  | ||||||
|         viewNode.addLight(sun); |  | ||||||
|         shRend.setLight(sun); |  | ||||||
|  |  | ||||||
|         final AmbientLight ambientLight = new AmbientLight(new ColorRGBA(0.3f, 0.3f, 0.3f, 0f)); |  | ||||||
|         viewNode.addLight(ambientLight); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Sets up the sky in the scene using a skybox with textures for all six directions. |  | ||||||
|      * This creates a realistic and immersive environment for the sea. |  | ||||||
|      */ |  | ||||||
|     private void setupSky() { |  | ||||||
|         final AssetManager assetManager = getApp().getAssetManager(); |  | ||||||
|         final Texture west = assetManager.loadTexture("Textures/Sky/Lagoon/lagoon_west.jpg"); //NON-NLS |  | ||||||
|         final Texture east = assetManager.loadTexture("Textures/Sky/Lagoon/lagoon_east.jpg"); //NON-NLS |  | ||||||
|         final Texture north = assetManager.loadTexture("Textures/Sky/Lagoon/lagoon_north.jpg"); //NON-NLS |  | ||||||
|         final Texture south = assetManager.loadTexture("Textures/Sky/Lagoon/lagoon_south.jpg"); //NON-NLS |  | ||||||
|         final Texture up = assetManager.loadTexture("Textures/Sky/Lagoon/lagoon_up.jpg"); //NON-NLS |  | ||||||
|         final Texture down = assetManager.loadTexture("Textures/Sky/Lagoon/lagoon_down.jpg"); //NON-NLS |  | ||||||
|         final Spatial sky = SkyFactory.createSky(assetManager, west, east, north, south, up, down); |  | ||||||
|         sky.rotate(0f, FloatMath.PI, 0f); |  | ||||||
|         viewNode.attachChild(sky); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Sets up the sea surface in the scene. This includes creating the sea mesh, |  | ||||||
|      * applying textures, and enabling shadows. |  | ||||||
|      */ |  | ||||||
|     private void setupScene() { |  | ||||||
|         final ShipMap ownMap = getGameLogic().getOwnMap(); |  | ||||||
|         final float x = 0.5f * ownMap.getWidth(); |  | ||||||
|         final float y = 0.5f * ownMap.getHeight(); |  | ||||||
|         final Box seaMesh = new Box(y + 0.5f, 0.1f, x + 0.5f); |  | ||||||
|         final Geometry seaGeo = new Geometry("sea", seaMesh); //NON-NLS |  | ||||||
|         seaGeo.setLocalTranslation(new Vector3f(y, -0.1f, x)); |  | ||||||
|         seaMesh.scaleTextureCoordinates(new Vector2f(4f, 4f)); |  | ||||||
|         final Material seaMat = getApp().getAssetManager().loadMaterial(SEA_TEXTURE); |  | ||||||
|         seaGeo.setMaterial(seaMat); |  | ||||||
|         seaGeo.setShadowMode(ShadowMode.CastAndReceive); |  | ||||||
|         TangentBinormalGenerator.generate(seaGeo); |  | ||||||
|         sceneNode.attachChild(seaGeo); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Adjusts the camera position and orientation to create a circular motion around |  | ||||||
|      * the center of the map. This provides a dynamic view of the sea and surrounding environment. |  | ||||||
|      */ |  | ||||||
|     private void adjustCamera() { |  | ||||||
|         final ShipMap ownMap = getGameLogic().getOwnMap(); |  | ||||||
|         final float mx = 0.5f * ownMap.getWidth(); |  | ||||||
|         final float my = 0.5f * ownMap.getHeight(); |  | ||||||
|         final float radius = 2f * sqrt(mx * mx + my + my); |  | ||||||
|         final float cos = radius * cos(cameraAngle); |  | ||||||
|         final float sin = radius * sin(cameraAngle); |  | ||||||
|         final float x = mx - cos; |  | ||||||
|         final float y = my - sin; |  | ||||||
|         final Camera camera = getApp().getCamera(); |  | ||||||
|         camera.setLocation(new Vector3f(y, ABOVE_SEA_LEVEL, x)); |  | ||||||
|         camera.getRotation().lookAt(new Vector3f(sin, -INCLINATION, cos), |  | ||||||
|                                     Vector3f.UNIT_Y); |  | ||||||
|         camera.update(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,213 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.client.gui; |  | ||||||
|  |  | ||||||
| import com.jme3.material.Material; |  | ||||||
| import com.jme3.material.RenderState.BlendMode; |  | ||||||
| import com.jme3.math.ColorRGBA; |  | ||||||
| import com.jme3.renderer.queue.RenderQueue.ShadowMode; |  | ||||||
| import com.jme3.scene.Geometry; |  | ||||||
| import com.jme3.scene.Node; |  | ||||||
| import com.jme3.scene.Spatial; |  | ||||||
| import com.jme3.scene.shape.Box; |  | ||||||
| import com.jme3.scene.shape.Cylinder; |  | ||||||
| import pp.battleship.client.BattleshipApp; |  | ||||||
| import pp.battleship.model.Battleship; |  | ||||||
| import pp.battleship.model.Rotation; |  | ||||||
| import pp.battleship.model.ShipMap; |  | ||||||
| import pp.battleship.model.Shot; |  | ||||||
|  |  | ||||||
| import static java.util.Objects.requireNonNull; |  | ||||||
| import static pp.util.FloatMath.HALF_PI; |  | ||||||
| import static pp.util.FloatMath.PI; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * The {@code SeaSynchronizer} class is responsible for synchronizing the graphical |  | ||||||
|  * representation of the ships and shots on the sea map with the underlying data model. |  | ||||||
|  * It extends the {@link ShipMapSynchronizer} to provide specific synchronization |  | ||||||
|  * logic for the sea map. |  | ||||||
|  */ |  | ||||||
| class SeaSynchronizer extends ShipMapSynchronizer { |  | ||||||
|     private static final String UNSHADED = "Common/MatDefs/Misc/Unshaded.j3md"; //NON-NLS |  | ||||||
|     private static final String KING_GEORGE_V_MODEL = "Models/KingGeorgeV/KingGeorgeV.j3o"; //NON-NLS |  | ||||||
|     private static final String COLOR = "Color"; //NON-NLS |  | ||||||
|     private static final String SHIP = "ship"; //NON-NLS |  | ||||||
|     private static final String SHOT = "shot"; //NON-NLS |  | ||||||
|     private static final ColorRGBA BOX_COLOR = ColorRGBA.Gray; |  | ||||||
|     private static final ColorRGBA SPLASH_COLOR = new ColorRGBA(0f, 0f, 1f, 0.4f); |  | ||||||
|     private static final ColorRGBA HIT_COLOR = new ColorRGBA(1f, 0f, 0f, 0.4f); |  | ||||||
|  |  | ||||||
|     private final ShipMap map; |  | ||||||
|     private final BattleshipApp app; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a {@code SeaSynchronizer} object with the specified application, root node, and ship map. |  | ||||||
|      * |  | ||||||
|      * @param app  the Battleship application |  | ||||||
|      * @param root the root node to which graphical elements will be attached |  | ||||||
|      * @param map  the ship map containing the ships and shots |  | ||||||
|      */ |  | ||||||
|     public SeaSynchronizer(BattleshipApp app, Node root, ShipMap map) { |  | ||||||
|         super(app.getGameLogic().getOwnMap(), root); |  | ||||||
|         this.app = app; |  | ||||||
|         this.map = map; |  | ||||||
|         addExisting(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Visits a {@link Shot} and creates a graphical representation of it. |  | ||||||
|      * If the shot is a hit, it attaches the representation to the ship node. |  | ||||||
|      * |  | ||||||
|      * @param shot the shot to be represented |  | ||||||
|      * @return the graphical representation of the shot, or null if the shot is a hit |  | ||||||
|      * and the representation has been attached to the ship node |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public Spatial visit(Shot shot) { |  | ||||||
|         return shot.isHit() ? handleHit(shot) : createCylinder(shot); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles a hit by attaching its representation to the node that |  | ||||||
|      * contains the ship model as a child so that it moves with the ship. |  | ||||||
|      * |  | ||||||
|      * @param shot a hit |  | ||||||
|      * @return always null to prevent the representation from being attached |  | ||||||
|      * to the items node as well |  | ||||||
|      */ |  | ||||||
|     private Spatial handleHit(Shot shot) { |  | ||||||
|         final Battleship ship = requireNonNull(map.findShipAt(shot), "Missing ship"); |  | ||||||
|         final Node shipNode = requireNonNull((Node) getSpatial(ship), "Missing ship node"); |  | ||||||
|  |  | ||||||
|         final Geometry representation = createCylinder(shot); |  | ||||||
|         representation.getLocalTranslation().subtractLocal(shipNode.getLocalTranslation()); |  | ||||||
|         shipNode.attachChild(representation); |  | ||||||
|  |  | ||||||
|         return null; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates a cylinder geometry representing the specified shot. |  | ||||||
|      * The appearance of the cylinder depends on whether the shot is a hit or a miss. |  | ||||||
|      * |  | ||||||
|      * @param shot the shot to be represented |  | ||||||
|      * @return the geometry representing the shot |  | ||||||
|      */ |  | ||||||
|     private Geometry createCylinder(Shot shot) { |  | ||||||
|         final ColorRGBA color = shot.isHit() ? HIT_COLOR : SPLASH_COLOR; |  | ||||||
|         final float height = shot.isHit() ? 1.2f : 0.1f; |  | ||||||
|  |  | ||||||
|         final Cylinder cylinder = new Cylinder(2, 20, 0.45f, height, true); |  | ||||||
|         final Geometry geometry = new Geometry(SHOT, cylinder); |  | ||||||
|  |  | ||||||
|         geometry.setMaterial(createColoredMaterial(color)); |  | ||||||
|         geometry.rotate(HALF_PI, 0f, 0f); |  | ||||||
|         // compute the center of the shot in world coordinates |  | ||||||
|         geometry.setLocalTranslation(shot.getY() + 0.5f, 0f, shot.getX() + 0.5f); |  | ||||||
|  |  | ||||||
|         return geometry; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Visits a {@link Battleship} and creates a graphical representation of it. |  | ||||||
|      * The representation is either a 3D model or a simple box depending on the |  | ||||||
|      * type of battleship. |  | ||||||
|      * |  | ||||||
|      * @param ship the battleship to be represented |  | ||||||
|      * @return the node containing the graphical representation of the battleship |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public Spatial visit(Battleship ship) { |  | ||||||
|         final Node node = new Node(SHIP); |  | ||||||
|         node.attachChild(createShip(ship)); |  | ||||||
|         // compute the center of the ship in world coordinates |  | ||||||
|         final float x = 0.5f * (ship.getMinY() + ship.getMaxY() + 1f); |  | ||||||
|         final float z = 0.5f * (ship.getMinX() + ship.getMaxX() + 1f); |  | ||||||
|         node.setLocalTranslation(x, 0f, z); |  | ||||||
|         node.addControl(new ShipControl(ship)); |  | ||||||
|         return node; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates the appropriate graphical representation of the specified battleship. |  | ||||||
|      * The representation is either a detailed model or a simple box based on the length of the ship. |  | ||||||
|      * |  | ||||||
|      * @param ship the battleship to be represented |  | ||||||
|      * @return the spatial representing the battleship |  | ||||||
|      */ |  | ||||||
|     private Spatial createShip(Battleship ship) { |  | ||||||
|         return ship.getLength() == 4 ? createBattleship(ship) : createBox(ship); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates a simple box to represent a battleship that is not of the "King George V" type. |  | ||||||
|      * |  | ||||||
|      * @param ship the battleship to be represented |  | ||||||
|      * @return the geometry representing the battleship as a box |  | ||||||
|      */ |  | ||||||
|     private Spatial createBox(Battleship ship) { |  | ||||||
|         final Box box = new Box(0.5f * (ship.getMaxY() - ship.getMinY()) + 0.3f, |  | ||||||
|                                 0.3f, |  | ||||||
|                                 0.5f * (ship.getMaxX() - ship.getMinX()) + 0.3f); |  | ||||||
|         final Geometry geometry = new Geometry(SHIP, box); |  | ||||||
|         geometry.setMaterial(createColoredMaterial(BOX_COLOR)); |  | ||||||
|         geometry.setShadowMode(ShadowMode.CastAndReceive); |  | ||||||
|  |  | ||||||
|         return geometry; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates a new {@link Material} with the specified color. |  | ||||||
|      * If the color includes transparency (i.e., alpha value less than 1), |  | ||||||
|      * the material's render state is set to use alpha blending, allowing for |  | ||||||
|      * semi-transparent rendering. |  | ||||||
|      * |  | ||||||
|      * @param color the {@link ColorRGBA} to be applied to the material. If the alpha value |  | ||||||
|      *              of the color is less than 1, the material will support transparency. |  | ||||||
|      * @return a {@link Material} instance configured with the specified color and, |  | ||||||
|      * if necessary, alpha blending enabled. |  | ||||||
|      */ |  | ||||||
|     private Material createColoredMaterial(ColorRGBA color) { |  | ||||||
|         final Material material = new Material(app.getAssetManager(), UNSHADED); |  | ||||||
|         if (color.getAlpha() < 1f) |  | ||||||
|             material.getAdditionalRenderState().setBlendMode(BlendMode.Alpha); |  | ||||||
|         material.setColor(COLOR, color); |  | ||||||
|         return material; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates a detailed 3D model to represent a "King George V" battleship. |  | ||||||
|      * |  | ||||||
|      * @param ship the battleship to be represented |  | ||||||
|      * @return the spatial representing the "King George V" battleship |  | ||||||
|      */ |  | ||||||
|     private Spatial createBattleship(Battleship ship) { |  | ||||||
|         final Spatial model = app.getAssetManager().loadModel(KING_GEORGE_V_MODEL); |  | ||||||
|  |  | ||||||
|         model.rotate(-HALF_PI, calculateRotationAngle(ship.getRot()), 0f); |  | ||||||
|         model.scale(1.48f); |  | ||||||
|         model.setShadowMode(ShadowMode.CastAndReceive); |  | ||||||
|  |  | ||||||
|         return model; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Calculates the rotation angle for the specified rotation. |  | ||||||
|      * |  | ||||||
|      * @param rot the rotation of the battleship |  | ||||||
|      * @return the rotation angle in radians |  | ||||||
|      */ |  | ||||||
|     private static float calculateRotationAngle(Rotation rot) { |  | ||||||
|         return switch (rot) { |  | ||||||
|             case RIGHT -> HALF_PI; |  | ||||||
|             case DOWN -> 0f; |  | ||||||
|             case LEFT -> -HALF_PI; |  | ||||||
|             case UP -> PI; |  | ||||||
|         }; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,106 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.client.gui; |  | ||||||
|  |  | ||||||
| import com.jme3.math.Quaternion; |  | ||||||
| import com.jme3.math.Vector3f; |  | ||||||
| import com.jme3.renderer.RenderManager; |  | ||||||
| import com.jme3.renderer.ViewPort; |  | ||||||
| import com.jme3.scene.control.AbstractControl; |  | ||||||
| import pp.battleship.model.Battleship; |  | ||||||
|  |  | ||||||
| import static pp.util.FloatMath.DEG_TO_RAD; |  | ||||||
| import static pp.util.FloatMath.TWO_PI; |  | ||||||
| import static pp.util.FloatMath.sin; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Controls the oscillating pitch motion of a battleship model in the game. |  | ||||||
|  * The ship oscillates to simulate a realistic movement on water, based on its orientation and length. |  | ||||||
|  */ |  | ||||||
| class ShipControl extends AbstractControl { |  | ||||||
|     /** |  | ||||||
|      * The axis of rotation for the ship's pitch (tilting forward and backward). |  | ||||||
|      */ |  | ||||||
|     private final Vector3f axis; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The duration of one complete oscillation cycle in seconds. |  | ||||||
|      */ |  | ||||||
|     private final float cycle; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The amplitude of the pitch oscillation in radians, determining how much the ship tilts. |  | ||||||
|      */ |  | ||||||
|     private final float amplitude; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * A quaternion representing the ship's current pitch rotation. |  | ||||||
|      */ |  | ||||||
|     private final Quaternion pitch = new Quaternion(); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The current time within the oscillation cycle, used to calculate the ship's pitch angle. |  | ||||||
|      */ |  | ||||||
|     private float time; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a new ShipControl instance for the specified Battleship. |  | ||||||
|      * The ship's orientation determines the axis of rotation, while its length influences |  | ||||||
|      * the cycle duration and amplitude of the oscillation. |  | ||||||
|      * |  | ||||||
|      * @param ship the Battleship object to control |  | ||||||
|      */ |  | ||||||
|     public ShipControl(Battleship ship) { |  | ||||||
|         // Determine the axis of rotation based on the ship's orientation |  | ||||||
|         axis = switch (ship.getRot()) { |  | ||||||
|             case LEFT, RIGHT -> Vector3f.UNIT_X; |  | ||||||
|             case UP, DOWN -> Vector3f.UNIT_Z; |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         // Set the cycle duration and amplitude based on the ship's length |  | ||||||
|         cycle = ship.getLength() * 2f; |  | ||||||
|         amplitude = 5f * DEG_TO_RAD / ship.getLength(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Updates the ship's pitch oscillation each frame. The ship's pitch is adjusted |  | ||||||
|      * to create a continuous tilting motion, simulating the effect of waves. |  | ||||||
|      * |  | ||||||
|      * @param tpf time per frame (in seconds), used to calculate the new pitch angle |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     protected void controlUpdate(float tpf) { |  | ||||||
|         // If spatial is null, do nothing |  | ||||||
|         if (spatial == null) return; |  | ||||||
|  |  | ||||||
|         // Update the time within the oscillation cycle |  | ||||||
|         time = (time + tpf) % cycle; |  | ||||||
|  |  | ||||||
|         // Calculate the current angle of the oscillation |  | ||||||
|         final float angle = amplitude * sin(time * TWO_PI / cycle); |  | ||||||
|  |  | ||||||
|         // Update the pitch Quaternion with the new angle |  | ||||||
|         pitch.fromAngleAxis(angle, axis); |  | ||||||
|  |  | ||||||
|         // Apply the pitch rotation to the spatial |  | ||||||
|         spatial.setLocalRotation(pitch); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * This method is called during the rendering phase, but it does not perform any |  | ||||||
|      * operations in this implementation as the control only influences the spatial's |  | ||||||
|      * transformation, not its rendering process. |  | ||||||
|      * |  | ||||||
|      * @param rm the RenderManager rendering the controlled Spatial (not null) |  | ||||||
|      * @param vp the ViewPort being rendered (not null) |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     protected void controlRender(RenderManager rm, ViewPort vp) { |  | ||||||
|         // No rendering logic is needed for this control |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,89 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.client.gui; |  | ||||||
|  |  | ||||||
| import com.jme3.scene.Node; |  | ||||||
| import com.jme3.scene.Spatial; |  | ||||||
| import pp.battleship.model.Item; |  | ||||||
| import pp.battleship.model.ShipMap; |  | ||||||
| import pp.battleship.model.Visitor; |  | ||||||
| import pp.battleship.notification.GameEventListener; |  | ||||||
| import pp.battleship.notification.ItemAddedEvent; |  | ||||||
| import pp.battleship.notification.ItemRemovedEvent; |  | ||||||
| import pp.view.ModelViewSynchronizer; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Abstract base class for synchronizing the visual representation of a {@link ShipMap} with its model state. |  | ||||||
|  * This class handles the addition and removal of items from the ship map, ensuring that changes in the model |  | ||||||
|  * are accurately reflected in the view. |  | ||||||
|  * <p> |  | ||||||
|  * Subclasses are responsible for providing the specific implementation of how each item in the map |  | ||||||
|  * is represented visually by implementing the {@link Visitor} interface. |  | ||||||
|  * </p> |  | ||||||
|  */ |  | ||||||
| abstract class ShipMapSynchronizer extends ModelViewSynchronizer<Item> implements Visitor<Spatial>, GameEventListener { |  | ||||||
|     // The ship map that this synchronizer is responsible for |  | ||||||
|     private final ShipMap shipMap; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a new ShipMapSynchronizer. |  | ||||||
|      * Initializes the synchronizer with the provided ship map and the root node for attaching view representations. |  | ||||||
|      * |  | ||||||
|      * @param map  the ship map to be synchronized |  | ||||||
|      * @param root the root node to which the view representations of the ship map items are attached |  | ||||||
|      */ |  | ||||||
|     protected ShipMapSynchronizer(ShipMap map, Node root) { |  | ||||||
|         super(root); |  | ||||||
|         this.shipMap = map; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Translates a model item into its corresponding visual representation. |  | ||||||
|      * The specific visual representation is determined by the concrete implementation of the {@link Visitor} interface. |  | ||||||
|      * |  | ||||||
|      * @param item the item from the model to be translated |  | ||||||
|      * @return the visual representation of the item as a {@link Spatial} |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     protected Spatial translate(Item item) { |  | ||||||
|         return item.accept(this); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Adds the existing items from the ship map to the view. |  | ||||||
|      * This method should be called during initialization to ensure that all current items in the ship map |  | ||||||
|      * are visually represented. |  | ||||||
|      */ |  | ||||||
|     protected void addExisting() { |  | ||||||
|         shipMap.getItems().forEach(this::add); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles the event when an item is removed from the ship map. |  | ||||||
|      * Removes the visual representation of the item from the view if it belongs to the synchronized ship map. |  | ||||||
|      * |  | ||||||
|      * @param event the event indicating that an item has been removed from the ship map |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void receivedEvent(ItemRemovedEvent event) { |  | ||||||
|         if (shipMap == event.map()) |  | ||||||
|             delete(event.item()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles the event when an item is added to the ship map. |  | ||||||
|      * Adds the visual representation of the new item to the view if it belongs to the synchronized ship map. |  | ||||||
|      * |  | ||||||
|      * @param event the event indicating that an item has been added to the ship map |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void receivedEvent(ItemAddedEvent event) { |  | ||||||
|         if (shipMap == event.map()) |  | ||||||
|             add(event.item()); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| Before Width: | Height: | Size: 235 KiB | 
| Before Width: | Height: | Size: 94 KiB | 
| @@ -1,3 +0,0 @@ | |||||||
| based on: |  | ||||||
| https://free3d.com/3d-model/wwii-ship-uk-king-george-v-class-battleship-v1--185381.html |  | ||||||
| License: Free Personal Use Only |  | ||||||
| @@ -1,4 +0,0 @@ | |||||||
| based on: |  | ||||||
| water 002 |  | ||||||
| by Katsukagi on 29/11/2018 |  | ||||||
| https://3dtextures.me/2018/11/29/water-002/ |  | ||||||
| @@ -1,14 +0,0 @@ | |||||||
| Material Water : Common/MatDefs/Light/Lighting.j3md { |  | ||||||
|      MaterialParameters { |  | ||||||
|          Shininess            : 64 |  | ||||||
|          DiffuseMap           : Repeat Textures/Terrain/Water/Water_002_COLOR.jpg |  | ||||||
|          NormalMap            : Repeat Textures/Terrain/Water/Water_002_NORM.jpg |  | ||||||
|          SpecularMap          : Repeat Textures/Terrain/Water/Water_002_ROUGH.jpg |  | ||||||
|          ParallaxMap          : Repeat Textures/Terrain/Water/Water_002_DISP.png |  | ||||||
|          // PackedNormalParallax : true |  | ||||||
|          // UseMaterialColors : true |  | ||||||
|          Ambient  : 0.5 0.5 0.5 1.0 |  | ||||||
|          Diffuse  : 1.0 1.0 1.0 1.0 |  | ||||||
|          Specular : 1.0 1.0 1.0 1.0 |  | ||||||
|      } |  | ||||||
| } |  | ||||||
| Before Width: | Height: | Size: 15 KiB | 
| Before Width: | Height: | Size: 207 KiB | 
| Before Width: | Height: | Size: 88 KiB | 
| Before Width: | Height: | Size: 31 KiB | 
| Before Width: | Height: | Size: 70 KiB | 
| @@ -1,16 +0,0 @@ | |||||||
| plugins { |  | ||||||
|     id 'buildlogic.jme-application-conventions' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| description = 'Battleship converter for resources' |  | ||||||
|  |  | ||||||
| dependencies { |  | ||||||
|     implementation libs.jme3.core |  | ||||||
|  |  | ||||||
|     runtimeOnly libs.jme3.desktop |  | ||||||
|     runtimeOnly libs.jme3.plugins |  | ||||||
| } |  | ||||||
|  |  | ||||||
| application { |  | ||||||
|     mainClass = 'pp.battleship.exporter.ModelExporter' |  | ||||||
| } |  | ||||||
| @@ -1,69 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.exporter; |  | ||||||
|  |  | ||||||
| import com.jme3.app.SimpleApplication; |  | ||||||
| import com.jme3.export.JmeExporter; |  | ||||||
| import com.jme3.export.binary.BinaryExporter; |  | ||||||
| import com.jme3.scene.Spatial; |  | ||||||
| import com.jme3.system.JmeContext; |  | ||||||
| import com.jme3.util.TangentBinormalGenerator; |  | ||||||
|  |  | ||||||
| import java.io.File; |  | ||||||
| import java.io.IOException; |  | ||||||
| import java.lang.System.Logger; |  | ||||||
| import java.lang.System.Logger.Level; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * This class transforms models into j3o format. |  | ||||||
|  */ |  | ||||||
| public class ModelExporter extends SimpleApplication { |  | ||||||
|     private static final Logger LOGGER = System.getLogger(ModelExporter.class.getName()); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The main method of the converter |  | ||||||
|      * |  | ||||||
|      * @param args input args |  | ||||||
|      */ |  | ||||||
|     public static void main(String[] args) { |  | ||||||
|         ModelExporter application = new ModelExporter(); |  | ||||||
|         application.start(JmeContext.Type.Headless); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Overrides  {@link com.jme3.app.SimpleApplication#simpleInitApp()}. |  | ||||||
|      * It initializes a simple app by exporting robots and rocks. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void simpleInitApp() { |  | ||||||
|         export("Models/KingGeorgeV/King_George_V.obj", "KingGeorgeV.j3o"); //NON-NLS |  | ||||||
|  |  | ||||||
|         stop(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Exports spatial into a file |  | ||||||
|      * |  | ||||||
|      * @param fileName       the file name of the model |  | ||||||
|      * @param exportFileName the name of the file where the .j3o file is going to be stored |  | ||||||
|      */ |  | ||||||
|     private void export(String fileName, String exportFileName) { |  | ||||||
|         final File file = new File(exportFileName); |  | ||||||
|         JmeExporter exporter = BinaryExporter.getInstance(); |  | ||||||
|         try { |  | ||||||
|             final Spatial model = getAssetManager().loadModel(fileName); |  | ||||||
|             TangentBinormalGenerator.generate(model); |  | ||||||
|             exporter.save(model, file); |  | ||||||
|         } |  | ||||||
|         catch (IOException exception) { |  | ||||||
|             LOGGER.log(Level.ERROR, "write to {0} failed", file); //NON-NLS |  | ||||||
|             throw new RuntimeException(exception); |  | ||||||
|         } |  | ||||||
|         LOGGER.log(Level.INFO, "wrote file {0}", file); //NON-NLS |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| Before Width: | Height: | Size: 235 KiB | 
| @@ -1,18 +0,0 @@ | |||||||
| # 3ds Max Wavefront OBJ Exporter v0.97b - (c)2007 guruware |  | ||||||
| # File Created: 16.03.2012 14:15:53 |  | ||||||
|  |  | ||||||
| newmtl _King_George_V |  | ||||||
| 	Ns 60.0000 |  | ||||||
| 	Ni 1.5000 |  | ||||||
| 	d 1.0000 |  | ||||||
| 	Tr 0.0000 |  | ||||||
| 	Tf 1.0000 1.0000 1.0000  |  | ||||||
| 	illum 2 |  | ||||||
| 	Ka 1.0000 1.0000 1.0000 |  | ||||||
| 	Kd 1.0000 1.0000 1.0000 |  | ||||||
| 	Ks 0.4500 0.4500 0.4500 |  | ||||||
| 	Ke 0.0000 0.0000 0.0000 |  | ||||||
| 	map_Ka King_George_V.jpg |  | ||||||
| 	map_Kd King_George_V.jpg |  | ||||||
| 	map_bump King_George_V_bump.jpg |  | ||||||
| 	bump King_George_V_bump.jpg |  | ||||||
| Before Width: | Height: | Size: 94 KiB | 
| @@ -1,3 +0,0 @@ | |||||||
| based on: |  | ||||||
| https://free3d.com/3d-model/wwii-ship-uk-king-george-v-class-battleship-v1--185381.html |  | ||||||
| License: Free Personal Use Only |  | ||||||
| @@ -1,12 +0,0 @@ | |||||||
| plugins { |  | ||||||
|     id 'buildlogic.java-library-conventions' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| description = 'Battleship common model' |  | ||||||
|  |  | ||||||
| dependencies { |  | ||||||
|     api project(":common") |  | ||||||
|     api libs.jme3.networking |  | ||||||
|     implementation libs.gson |  | ||||||
|     testImplementation libs.mockito.core |  | ||||||
| } |  | ||||||
| @@ -1,103 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship; |  | ||||||
|  |  | ||||||
| import pp.util.config.Config; |  | ||||||
|  |  | ||||||
| import java.util.Map; |  | ||||||
| import java.util.TreeMap; |  | ||||||
|  |  | ||||||
| import static java.lang.Math.max; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Provides access to the configuration settings for the Battleship game. |  | ||||||
|  * <p> |  | ||||||
|  * This class allows for loading configuration settings from a properties file, |  | ||||||
|  * including the server port, map dimensions, and the number of ships of various lengths. |  | ||||||
|  * </p> |  | ||||||
|  * <p> |  | ||||||
|  * <b>Note:</b> Attributes of this class are not marked as {@code final} to allow |  | ||||||
|  * for proper initialization when reading from a properties file. |  | ||||||
|  * </p> |  | ||||||
|  */ |  | ||||||
| public class BattleshipConfig extends Config { |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The default port number for the Battleship server. |  | ||||||
|      */ |  | ||||||
|     @Property("port") |  | ||||||
|     private int port = 1234; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The width of the game map in terms of grid units. |  | ||||||
|      */ |  | ||||||
|     @Property("map.width") |  | ||||||
|     private int mapWidth = 10; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The height of the game map in terms of grid units. |  | ||||||
|      */ |  | ||||||
|     @Property("map.height") |  | ||||||
|     private int mapHeight = 10; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * An array representing the number of ships available for each length. |  | ||||||
|      * The index corresponds to the ship length minus one, and the value at each index |  | ||||||
|      * is the number of ships of that length. |  | ||||||
|      */ |  | ||||||
|     @Property("ship.nums") |  | ||||||
|     private int[] shipNums = {4, 3, 2, 1}; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates an instance of {@code BattleshipConfig} with default settings. |  | ||||||
|      */ |  | ||||||
|     public BattleshipConfig() { |  | ||||||
|         // Default constructor |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the port number configured for the Battleship server. |  | ||||||
|      * |  | ||||||
|      * @return the port number |  | ||||||
|      */ |  | ||||||
|     public int getPort() { |  | ||||||
|         return port; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the width of the game map. The width is guaranteed to be at least 2 units. |  | ||||||
|      * |  | ||||||
|      * @return the width of the game map |  | ||||||
|      */ |  | ||||||
|     public int getMapWidth() { |  | ||||||
|         return max(mapWidth, 2); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the height of the game map. The height is guaranteed to be at least 2 units. |  | ||||||
|      * |  | ||||||
|      * @return the height of the game map |  | ||||||
|      */ |  | ||||||
|     public int getMapHeight() { |  | ||||||
|         return max(mapHeight, 2); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns a map representing the number of ships for each length. |  | ||||||
|      * The keys are ship lengths, and the values are the corresponding number of ships. |  | ||||||
|      * |  | ||||||
|      * @return a map of ship lengths to the number of ships |  | ||||||
|      */ |  | ||||||
|     public Map<Integer, Integer> getShipNums() { |  | ||||||
|         final TreeMap<Integer, Integer> ships = new TreeMap<>(); |  | ||||||
|         for (int i = 0; i < shipNums.length; i++) |  | ||||||
|             if (shipNums[i] > 0) |  | ||||||
|                 ships.put(i + 1, shipNums[i]); |  | ||||||
|         return ships; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,40 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship; |  | ||||||
|  |  | ||||||
| import java.util.ResourceBundle; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Provides access to the resource bundle of the game. |  | ||||||
|  * |  | ||||||
|  * @see #BUNDLE |  | ||||||
|  */ |  | ||||||
| public class Resources { |  | ||||||
|     /** |  | ||||||
|      * The resource bundle for the Battleship game. |  | ||||||
|      */ |  | ||||||
|     public static final ResourceBundle BUNDLE = ResourceBundle.getBundle("battleship"); //NON-NLS |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Gets a string for the given key from the resource bundle in {@linkplain #BUNDLE}. |  | ||||||
|      * |  | ||||||
|      * @param key the key for the desired string |  | ||||||
|      * @return the string for the given key |  | ||||||
|      * @throws NullPointerException               if {@code key} is {@code null} |  | ||||||
|      * @throws java.util.MissingResourceException if no object for the given key can be found |  | ||||||
|      * @throws ClassCastException                 if the object found for the given key is not a string |  | ||||||
|      */ |  | ||||||
|     public static String lookup(String key) { |  | ||||||
|         return BUNDLE.getString(key); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Private constructor to prevent instantiation. |  | ||||||
|      */ |  | ||||||
|     private Resources() { /* do not instantiate */ } |  | ||||||
| } |  | ||||||
| @@ -1,102 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.game.client; |  | ||||||
|  |  | ||||||
| import pp.battleship.message.client.ShootMessage; |  | ||||||
| import pp.battleship.message.server.EffectMessage; |  | ||||||
| import pp.battleship.model.IntPoint; |  | ||||||
| import pp.battleship.model.ShipMap; |  | ||||||
| import pp.battleship.notification.Sound; |  | ||||||
|  |  | ||||||
| import java.lang.System.Logger.Level; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Represents the state of the client where players take turns to attack each other's ships. |  | ||||||
|  */ |  | ||||||
| class BattleState extends ClientState { |  | ||||||
|     private boolean myTurn; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a new instance of {@link BattleState}. |  | ||||||
|      * |  | ||||||
|      * @param logic  the game logic |  | ||||||
|      * @param myTurn true if it is my turn |  | ||||||
|      */ |  | ||||||
|     public BattleState(ClientGameLogic logic, boolean myTurn) { |  | ||||||
|         super(logic); |  | ||||||
|         this.myTurn = myTurn; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public boolean showBattle() { |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public void clickOpponentMap(IntPoint pos) { |  | ||||||
|         if (!myTurn) |  | ||||||
|             logic.setInfoText("wait.its.not.your.turn"); |  | ||||||
|         else if (logic.getOpponentMap().isValid(pos)) |  | ||||||
|             logic.send(new ShootMessage(pos)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Reports the effect of a shot based on the server message. |  | ||||||
|      * |  | ||||||
|      * @param msg the message containing the effect of the shot |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void receivedEffect(EffectMessage msg) { |  | ||||||
|         ClientGameLogic.LOGGER.log(Level.INFO, "report effect: {0}", msg); //NON-NLS |  | ||||||
|         playSound(msg); |  | ||||||
|         myTurn = msg.isMyTurn(); |  | ||||||
|         logic.setInfoText(msg.getInfoTextKey()); |  | ||||||
|         affectedMap(msg).add(msg.getShot()); |  | ||||||
|         if (destroyedOpponentShip(msg)) |  | ||||||
|             logic.getOpponentMap().add(msg.getDestroyedShip()); |  | ||||||
|         if (msg.isGameOver()) { |  | ||||||
|             msg.getRemainingOpponentShips().forEach(logic.getOwnMap()::add); |  | ||||||
|             logic.setState(new GameOverState(logic)); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Determines which map (own or opponent's) should be affected by the shot based on the message. |  | ||||||
|      * |  | ||||||
|      * @param msg the effect message received from the server |  | ||||||
|      * @return the map (either the opponent's or player's own map) that is affected by the shot |  | ||||||
|      */ |  | ||||||
|     private ShipMap affectedMap(EffectMessage msg) { |  | ||||||
|         return msg.isOwnShot() ? logic.getOpponentMap() : logic.getOwnMap(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the opponent's ship was destroyed by the player's shot. |  | ||||||
|      * |  | ||||||
|      * @param msg the effect message received from the server |  | ||||||
|      * @return true if the shot destroyed an opponent's ship, false otherwise |  | ||||||
|      */ |  | ||||||
|     private boolean destroyedOpponentShip(EffectMessage msg) { |  | ||||||
|         return msg.getDestroyedShip() != null && msg.isOwnShot(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Plays a sound based on the outcome of the shot. Different sounds are played for a miss, hit, |  | ||||||
|      * or destruction of a ship. |  | ||||||
|      * |  | ||||||
|      * @param msg the effect message containing the result of the shot |  | ||||||
|      */ |  | ||||||
|     private void playSound(EffectMessage msg) { |  | ||||||
|         if (!msg.getShot().isHit()) |  | ||||||
|             logic.playSound(Sound.SPLASH); |  | ||||||
|         else if (msg.getDestroyedShip() == null) |  | ||||||
|             logic.playSound(Sound.EXPLOSION); |  | ||||||
|         else |  | ||||||
|             logic.playSound(Sound.DESTROYED_SHIP); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,38 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.game.client; |  | ||||||
|  |  | ||||||
| import pp.battleship.game.singlemode.BattleshipClientConfig; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Interface representing a Battleship client. |  | ||||||
|  * Provides methods to access game logic, configuration, and to enqueue tasks. |  | ||||||
|  */ |  | ||||||
| public interface BattleshipClient { |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the game logic associated with this client. |  | ||||||
|      * |  | ||||||
|      * @return the ClientGameLogic instance |  | ||||||
|      */ |  | ||||||
|     ClientGameLogic getGameLogic(); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the configuration associated with this client. |  | ||||||
|      * |  | ||||||
|      * @return the BattleshipConfig instance |  | ||||||
|      */ |  | ||||||
|     BattleshipClientConfig getConfig(); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Enqueues a task to be executed by the client. |  | ||||||
|      * |  | ||||||
|      * @param runnable the task to be executed |  | ||||||
|      */ |  | ||||||
|     void enqueue(Runnable runnable); |  | ||||||
| } |  | ||||||
| @@ -1,355 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.game.client; |  | ||||||
|  |  | ||||||
| import pp.battleship.message.client.ClientMessage; |  | ||||||
| import pp.battleship.message.server.EffectMessage; |  | ||||||
| import pp.battleship.message.server.GameDetails; |  | ||||||
| import pp.battleship.message.server.ServerInterpreter; |  | ||||||
| import pp.battleship.message.server.StartBattleMessage; |  | ||||||
| import pp.battleship.model.IntPoint; |  | ||||||
| import pp.battleship.model.ShipMap; |  | ||||||
| import pp.battleship.model.dto.ShipMapDTO; |  | ||||||
| import pp.battleship.notification.ClientStateEvent; |  | ||||||
| import pp.battleship.notification.GameEvent; |  | ||||||
| import pp.battleship.notification.GameEventBroker; |  | ||||||
| import pp.battleship.notification.GameEventListener; |  | ||||||
| import pp.battleship.notification.InfoTextEvent; |  | ||||||
| import pp.battleship.notification.Sound; |  | ||||||
| import pp.battleship.notification.SoundEvent; |  | ||||||
|  |  | ||||||
| import java.io.File; |  | ||||||
| import java.io.IOException; |  | ||||||
| import java.lang.System.Logger; |  | ||||||
| import java.lang.System.Logger.Level; |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| import static java.lang.Math.max; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Controls the client-side game logic for Battleship. |  | ||||||
|  * Manages the player's ship placement, interactions with the map, and response to server messages. |  | ||||||
|  */ |  | ||||||
| public class ClientGameLogic implements ServerInterpreter, GameEventBroker { |  | ||||||
|     static final Logger LOGGER = System.getLogger(ClientGameLogic.class.getName()); |  | ||||||
|     private final ClientSender clientSender; |  | ||||||
|     private final List<GameEventListener> listeners = new ArrayList<>(); |  | ||||||
|     private GameDetails details; |  | ||||||
|     private ShipMap ownMap; |  | ||||||
|     private ShipMap harbor; |  | ||||||
|     private ShipMap opponentMap; |  | ||||||
|     private ClientState state = new InitialState(this); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a ClientGameLogic with the specified sender object. |  | ||||||
|      * |  | ||||||
|      * @param clientSender the object used to send messages to the server |  | ||||||
|      */ |  | ||||||
|     public ClientGameLogic(ClientSender clientSender) { |  | ||||||
|         this.clientSender = clientSender; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the current state of the game logic. |  | ||||||
|      */ |  | ||||||
|     ClientState getState() { |  | ||||||
|         return state; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Sets the current state of the game logic. |  | ||||||
|      * |  | ||||||
|      * @param newState the new state to be set |  | ||||||
|      */ |  | ||||||
|     void setState(ClientState newState) { |  | ||||||
|         LOGGER.log(Level.DEBUG, "state transition {0} --> {1}", state.getName(), newState.getName()); //NON-NLS |  | ||||||
|         state = newState; |  | ||||||
|         notifyListeners(new ClientStateEvent()); |  | ||||||
|         state.entry(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the game details. |  | ||||||
|      * |  | ||||||
|      * @return the game details |  | ||||||
|      */ |  | ||||||
|     GameDetails getDetails() { |  | ||||||
|         return details; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the player's own map. |  | ||||||
|      * |  | ||||||
|      * @return the player's own map |  | ||||||
|      */ |  | ||||||
|     public ShipMap getOwnMap() { |  | ||||||
|         return ownMap; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the opponent's map. |  | ||||||
|      * |  | ||||||
|      * @return the opponent's map |  | ||||||
|      */ |  | ||||||
|     public ShipMap getOpponentMap() { |  | ||||||
|         return opponentMap; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the harbor map. |  | ||||||
|      * |  | ||||||
|      * @return the harbor map |  | ||||||
|      */ |  | ||||||
|     public ShipMap getHarbor() { |  | ||||||
|         return harbor; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the editor should be shown. |  | ||||||
|      * |  | ||||||
|      * @return true if the editor should be shown, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean showEditor() { |  | ||||||
|         return state.showEditor(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the battle state should be shown. |  | ||||||
|      * |  | ||||||
|      * @return true if the battle state should be shown, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean showBattle() { |  | ||||||
|         return state.showBattle(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Sets the game details provided by the server. |  | ||||||
|      * |  | ||||||
|      * @param details the game details including map size and ships |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void received(GameDetails details) { |  | ||||||
|         state.receivedGameDetails(details); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Moves the preview ship to the specified position. |  | ||||||
|      * |  | ||||||
|      * @param pos the new position for the preview ship |  | ||||||
|      */ |  | ||||||
|     public void movePreview(IntPoint pos) { |  | ||||||
|         state.movePreview(pos); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles a click on the player's own map. |  | ||||||
|      * |  | ||||||
|      * @param pos the position where the click occurred |  | ||||||
|      */ |  | ||||||
|     public void clickOwnMap(IntPoint pos) { |  | ||||||
|         state.clickOwnMap(pos); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles a click on the harbor map. |  | ||||||
|      * |  | ||||||
|      * @param pos the position where the click occurred |  | ||||||
|      */ |  | ||||||
|     public void clickHarbor(IntPoint pos) { |  | ||||||
|         state.clickHarbor(pos); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles a click on the opponent's map. |  | ||||||
|      * |  | ||||||
|      * @param pos the position where the click occurred |  | ||||||
|      */ |  | ||||||
|     public void clickOpponentMap(IntPoint pos) { |  | ||||||
|         state.clickOpponentMap(pos); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Rotates the preview ship. |  | ||||||
|      */ |  | ||||||
|     public void rotateShip() { |  | ||||||
|         state.rotateShip(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Marks the player's map as finished. |  | ||||||
|      */ |  | ||||||
|     public void mapFinished() { |  | ||||||
|         state.mapFinished(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the player's map is complete (i.e., all ships are placed). |  | ||||||
|      * |  | ||||||
|      * @return true if all ships are placed, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean isMapComplete() { |  | ||||||
|         return state.isMapComplete(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if there is currently a preview ship. |  | ||||||
|      * |  | ||||||
|      * @return true if there is currently a preview ship, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean movingShip() { |  | ||||||
|         return state.movingShip(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Starts the battle based on the server message. |  | ||||||
|      * |  | ||||||
|      * @param msg the message indicating whose turn it is to shoot |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void received(StartBattleMessage msg) { |  | ||||||
|         state.receivedStartBattle(msg); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Reports the effect of a shot based on the server message. |  | ||||||
|      * |  | ||||||
|      * @param msg the message containing the effect of the shot |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void received(EffectMessage msg) { |  | ||||||
|         state.receivedEffect(msg); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Initializes the player's own map, opponent's map, and harbor based on the game details. |  | ||||||
|      * |  | ||||||
|      * @param details the game details including map size and ships |  | ||||||
|      */ |  | ||||||
|     void initializeMaps(GameDetails details) { |  | ||||||
|         this.details = details; |  | ||||||
|         final int numShips = details.getShipNums().values().stream().mapToInt(Integer::intValue).sum(); |  | ||||||
|         final int maxLength = details.getShipNums().keySet().stream().mapToInt(Integer::intValue).max().orElse(2); |  | ||||||
|         ownMap = new ShipMap(details.getWidth(), details.getHeight(), this); |  | ||||||
|         opponentMap = new ShipMap(details.getWidth(), details.getHeight(), this); |  | ||||||
|         harbor = new ShipMap(max(maxLength, 2), max(numShips, details.getHeight()), this); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Sets the informational text to be displayed to the player. |  | ||||||
|      * |  | ||||||
|      * @param key the key for the info text |  | ||||||
|      */ |  | ||||||
|     void setInfoText(String key) { |  | ||||||
|         notifyListeners(new InfoTextEvent(key)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Emits an event to play the specified sound. |  | ||||||
|      * |  | ||||||
|      * @param sound the sound to be played. |  | ||||||
|      */ |  | ||||||
|     public void playSound(Sound sound) { |  | ||||||
|         notifyListeners(new SoundEvent(sound)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Loads a map from the specified file. |  | ||||||
|      * |  | ||||||
|      * @param file the file to load the map from |  | ||||||
|      * @throws IOException if an I/O error occurs |  | ||||||
|      */ |  | ||||||
|     public void loadMap(File file) throws IOException { |  | ||||||
|         state.loadMap(file); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the player's own map may be loaded from a file. |  | ||||||
|      * |  | ||||||
|      * @return true if the own map may be loaded from file, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean mayLoadMap() { |  | ||||||
|         return state.mayLoadMap(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the player's own map may be saved to a file. |  | ||||||
|      * |  | ||||||
|      * @return true if the own map may be saved to file, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean maySaveMap() { |  | ||||||
|         return state.maySaveMap(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Saves the player's own map to the specified file. |  | ||||||
|      * |  | ||||||
|      * @param file the file to save the map to |  | ||||||
|      * @throws IOException if the map cannot be saved in the current state |  | ||||||
|      */ |  | ||||||
|     public void saveMap(File file) throws IOException { |  | ||||||
|         if (ownMap != null && maySaveMap()) |  | ||||||
|             new ShipMapDTO(ownMap).saveTo(file); |  | ||||||
|         else |  | ||||||
|             throw new IOException("You are not allowed to save the map in this state of the game"); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Sends a message to the server. |  | ||||||
|      * |  | ||||||
|      * @param msg the message to be sent |  | ||||||
|      */ |  | ||||||
|     void send(ClientMessage msg) { |  | ||||||
|         if (clientSender == null) |  | ||||||
|             LOGGER.log(Level.ERROR, "trying to send {0} with sender==null", msg); //NON-NLS |  | ||||||
|         else |  | ||||||
|             clientSender.send(msg); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Adds a listener to receive game events. |  | ||||||
|      * |  | ||||||
|      * @param listener the listener to add |  | ||||||
|      */ |  | ||||||
|     public synchronized void addListener(GameEventListener listener) { |  | ||||||
|         listeners.add(listener); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Removes a listener from receiving game events. |  | ||||||
|      * |  | ||||||
|      * @param listener the listener to remove |  | ||||||
|      */ |  | ||||||
|     public synchronized void removeListener(GameEventListener listener) { |  | ||||||
|         listeners.remove(listener); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Notifies all listeners of a game event. |  | ||||||
|      * |  | ||||||
|      * @param event the game event to notify listeners of |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void notifyListeners(GameEvent event) { |  | ||||||
|         final List<GameEventListener> copy; |  | ||||||
|         synchronized (this) { |  | ||||||
|             copy = new ArrayList<>(listeners); |  | ||||||
|         } |  | ||||||
|         for (GameEventListener listener : copy) |  | ||||||
|             event.notifyListener(listener); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Called once per frame by the update loop. |  | ||||||
|      * |  | ||||||
|      * @param delta time in seconds since the last update call |  | ||||||
|      */ |  | ||||||
|     public void update(float delta) { |  | ||||||
|         state.update(delta); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,22 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.game.client; |  | ||||||
|  |  | ||||||
| import pp.battleship.message.client.ClientMessage; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Interface for sending messages to the server. |  | ||||||
|  */ |  | ||||||
| public interface ClientSender { |  | ||||||
|     /** |  | ||||||
|      * Send the specified message to the server. |  | ||||||
|      * |  | ||||||
|      * @param message the message |  | ||||||
|      */ |  | ||||||
|     void send(ClientMessage message); |  | ||||||
| } |  | ||||||
| @@ -1,202 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.game.client; |  | ||||||
|  |  | ||||||
| import pp.battleship.message.server.EffectMessage; |  | ||||||
| import pp.battleship.message.server.GameDetails; |  | ||||||
| import pp.battleship.message.server.StartBattleMessage; |  | ||||||
| import pp.battleship.model.IntPoint; |  | ||||||
|  |  | ||||||
| import java.io.File; |  | ||||||
| import java.io.IOException; |  | ||||||
| import java.lang.System.Logger.Level; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Defines the behavior and state transitions for the client-side game logic. |  | ||||||
|  * Different states of the game logic implement this interface to handle various game events and actions. |  | ||||||
|  */ |  | ||||||
| abstract class ClientState { |  | ||||||
|     /** |  | ||||||
|      * The game logic object. |  | ||||||
|      */ |  | ||||||
|     final ClientGameLogic logic; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a client state of the specified game logic. |  | ||||||
|      * |  | ||||||
|      * @param logic the game logic |  | ||||||
|      */ |  | ||||||
|     ClientState(ClientGameLogic logic) { |  | ||||||
|         this.logic = logic; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Method to be overridden by subclasses for post-transition initialization. |  | ||||||
|      * By default, it does nothing, but it can be overridden in derived states. |  | ||||||
|      */ |  | ||||||
|     void entry() { |  | ||||||
|         // Default implementation does nothing |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the name of the current state. |  | ||||||
|      * |  | ||||||
|      * @return the name of the current state |  | ||||||
|      */ |  | ||||||
|     String getName() { |  | ||||||
|         return getClass().getSimpleName(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the editor should be shown. |  | ||||||
|      * |  | ||||||
|      * @return true if the editor should be shown, false otherwise |  | ||||||
|      */ |  | ||||||
|     boolean showEditor() { |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the battle state should be shown. |  | ||||||
|      * |  | ||||||
|      * @return true if the battle state should be shown, false otherwise |  | ||||||
|      */ |  | ||||||
|     boolean showBattle() { |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the player's map is complete (i.e., all ships are placed). |  | ||||||
|      * |  | ||||||
|      * @return true if all ships are placed, false otherwise |  | ||||||
|      */ |  | ||||||
|     boolean isMapComplete() { |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if there is currently a preview ship. |  | ||||||
|      * |  | ||||||
|      * @return true if there is currently a preview ship, false otherwise |  | ||||||
|      */ |  | ||||||
|     boolean movingShip() { |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles a click on the player's own map. |  | ||||||
|      * |  | ||||||
|      * @param pos the position where the click occurred |  | ||||||
|      */ |  | ||||||
|     void clickOwnMap(IntPoint pos) { |  | ||||||
|         ClientGameLogic.LOGGER.log(Level.DEBUG, "clickOwnMap has no effect in {0}", getName()); //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles a click on the harbor map. |  | ||||||
|      * |  | ||||||
|      * @param pos the position where the click occurred |  | ||||||
|      */ |  | ||||||
|     void clickHarbor(IntPoint pos) { |  | ||||||
|         ClientGameLogic.LOGGER.log(Level.DEBUG, "clickHarbor has no effect in {0}", getName()); //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles a click on the opponent's map. |  | ||||||
|      * |  | ||||||
|      * @param pos the position where the click occurred |  | ||||||
|      */ |  | ||||||
|     void clickOpponentMap(IntPoint pos) { |  | ||||||
|         ClientGameLogic.LOGGER.log(Level.DEBUG, "clickOpponentMap has no effect in {0}", getName()); //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Moves the preview ship to the specified position. |  | ||||||
|      * |  | ||||||
|      * @param pos the new position for the preview ship |  | ||||||
|      */ |  | ||||||
|     void movePreview(IntPoint pos) { |  | ||||||
|         ClientGameLogic.LOGGER.log(Level.DEBUG, "movePreview has no effect in {0}", getName()); //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Rotates the preview ship. |  | ||||||
|      */ |  | ||||||
|     void rotateShip() { |  | ||||||
|         ClientGameLogic.LOGGER.log(Level.DEBUG, "rotateShip has no effect in {0}", getName()); //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The user has marked the map as finished. |  | ||||||
|      */ |  | ||||||
|     void mapFinished() { |  | ||||||
|         ClientGameLogic.LOGGER.log(Level.ERROR, "mapFinished not allowed in {0}", getName()); //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Sets the game details provided by the server. |  | ||||||
|      * |  | ||||||
|      * @param details the game details including map size and ships |  | ||||||
|      */ |  | ||||||
|     void receivedGameDetails(GameDetails details) { |  | ||||||
|         ClientGameLogic.LOGGER.log(Level.ERROR, "receivedGameDetails not allowed in {0}", getName()); //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Starts the battle based on the server message. |  | ||||||
|      * |  | ||||||
|      * @param msg the message indicating whose turn it is to shoot |  | ||||||
|      */ |  | ||||||
|     void receivedStartBattle(StartBattleMessage msg) { |  | ||||||
|         ClientGameLogic.LOGGER.log(Level.ERROR, "receivedStartBattle not allowed in {0}", getName()); //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Reports the effect of a shot based on the server message. |  | ||||||
|      * |  | ||||||
|      * @param msg the message containing the effect of the shot |  | ||||||
|      */ |  | ||||||
|     void receivedEffect(EffectMessage msg) { |  | ||||||
|         ClientGameLogic.LOGGER.log(Level.ERROR, "receivedEffect not allowed in {0}", getName()); //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Loads a map from the specified file. |  | ||||||
|      * |  | ||||||
|      * @param file the file to load the map from |  | ||||||
|      * @throws IOException if the map cannot be loaded in the current state |  | ||||||
|      */ |  | ||||||
|     void loadMap(File file) throws IOException { |  | ||||||
|         throw new IOException("You are not allowed to load a map in this state of the game"); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the own map may be loaded from file. |  | ||||||
|      * |  | ||||||
|      * @return true if the own map may be loaded from file, false otherwise |  | ||||||
|      */ |  | ||||||
|     boolean mayLoadMap() { |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the own map may be saved to file. |  | ||||||
|      * |  | ||||||
|      * @return true if the own map may be saved to file, false otherwise |  | ||||||
|      */ |  | ||||||
|     boolean maySaveMap() { |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Called once per frame by the update loop if this state is active. |  | ||||||
|      * |  | ||||||
|      * @param delta time in seconds since the last update call |  | ||||||
|      */ |  | ||||||
|     void update(float delta) { /* do nothing by default */ } |  | ||||||
| } |  | ||||||
| @@ -1,267 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.game.client; |  | ||||||
|  |  | ||||||
| import pp.battleship.message.client.MapMessage; |  | ||||||
| import pp.battleship.model.Battleship; |  | ||||||
| import pp.battleship.model.IntPoint; |  | ||||||
| import pp.battleship.model.ShipMap; |  | ||||||
| import pp.battleship.model.dto.ShipMapDTO; |  | ||||||
|  |  | ||||||
| import java.io.File; |  | ||||||
| import java.io.IOException; |  | ||||||
| import java.lang.System.Logger.Level; |  | ||||||
|  |  | ||||||
| import static pp.battleship.Resources.lookup; |  | ||||||
| import static pp.battleship.model.Battleship.Status.INVALID_PREVIEW; |  | ||||||
| import static pp.battleship.model.Battleship.Status.NORMAL; |  | ||||||
| import static pp.battleship.model.Battleship.Status.VALID_PREVIEW; |  | ||||||
| import static pp.battleship.model.Rotation.RIGHT; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Represents the state of the client setting up the ship map. |  | ||||||
|  */ |  | ||||||
| class EditorState extends ClientState { |  | ||||||
|     private Battleship preview; |  | ||||||
|     private Battleship selectedInHarbor; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a new EditorState with the specified ClientGameLogic. |  | ||||||
|      * |  | ||||||
|      * @param logic the ClientGameLogic associated with this state |  | ||||||
|      */ |  | ||||||
|     public EditorState(ClientGameLogic logic) { |  | ||||||
|         super(logic); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns true to indicate that the editor should be shown. |  | ||||||
|      * |  | ||||||
|      * @return true if the editor should be shown |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public boolean showEditor() { |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Moves the preview ship to the specified position. |  | ||||||
|      * |  | ||||||
|      * @param pos the new position for the preview ship |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void movePreview(IntPoint pos) { |  | ||||||
|         ClientGameLogic.LOGGER.log(Level.DEBUG, "move preview to {0}", pos); //NON-NLS |  | ||||||
|         if (preview == null || !ownMap().isValid(pos)) return; |  | ||||||
|         preview.moveTo(pos); |  | ||||||
|         setPreviewStatus(preview); |  | ||||||
|         ownMap().remove(preview); |  | ||||||
|         ownMap().add(preview); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles a click on the player's own map. |  | ||||||
|      * |  | ||||||
|      * @param pos the position where the click occurred |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void clickOwnMap(IntPoint pos) { |  | ||||||
|         ClientGameLogic.LOGGER.log(Level.DEBUG, "click at {0} in own map", pos); //NON-NLS |  | ||||||
|         if (!ownMap().isValid(pos)) return; |  | ||||||
|         if (preview == null) |  | ||||||
|             modifyShip(pos); |  | ||||||
|         else |  | ||||||
|             placeShip(pos); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Modifies a ship on the map at the specified position. |  | ||||||
|      * |  | ||||||
|      * @param cursor the position of the ship to modify |  | ||||||
|      */ |  | ||||||
|     private void modifyShip(IntPoint cursor) { |  | ||||||
|         preview = ownMap().findShipAt(cursor); |  | ||||||
|         if (preview == null) |  | ||||||
|             return; |  | ||||||
|         preview.moveTo(cursor); |  | ||||||
|         setPreviewStatus(preview); |  | ||||||
|         ownMap().remove(preview); |  | ||||||
|         ownMap().add(preview); |  | ||||||
|         selectedInHarbor = new Battleship(preview.getLength(), 0, freeY(), RIGHT); |  | ||||||
|         selectedInHarbor.setStatus(VALID_PREVIEW); |  | ||||||
|         harbor().add(selectedInHarbor); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Places the preview ship at the specified position. |  | ||||||
|      * |  | ||||||
|      * @param cursor the position to place the ship |  | ||||||
|      */ |  | ||||||
|     private void placeShip(IntPoint cursor) { |  | ||||||
|         ownMap().remove(preview); |  | ||||||
|         preview.moveTo(cursor); |  | ||||||
|         if (ownMap().isValid(preview)) { |  | ||||||
|             preview.setStatus(NORMAL); |  | ||||||
|             ownMap().add(preview); |  | ||||||
|             harbor().remove(selectedInHarbor); |  | ||||||
|             preview = null; |  | ||||||
|             selectedInHarbor = null; |  | ||||||
|         } |  | ||||||
|         else { |  | ||||||
|             preview.setStatus(INVALID_PREVIEW); |  | ||||||
|             ownMap().add(preview); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles a click on the harbor map. |  | ||||||
|      * |  | ||||||
|      * @param pos the position where the click occurred |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void clickHarbor(IntPoint pos) { |  | ||||||
|         ClientGameLogic.LOGGER.log(Level.DEBUG, "click at {0} in harbor", pos); //NON-NLS |  | ||||||
|         if (!harbor().isValid(pos)) return; |  | ||||||
|         final Battleship shipAtCursor = harbor().findShipAt(pos); |  | ||||||
|         if (preview != null) { |  | ||||||
|             ownMap().remove(preview); |  | ||||||
|             selectedInHarbor.setStatus(NORMAL); |  | ||||||
|             harbor().remove(selectedInHarbor); |  | ||||||
|             harbor().add(selectedInHarbor); |  | ||||||
|             preview = null; |  | ||||||
|             selectedInHarbor = null; |  | ||||||
|         } |  | ||||||
|         else if (shipAtCursor != null) { |  | ||||||
|             selectedInHarbor = shipAtCursor; |  | ||||||
|             selectedInHarbor.setStatus(VALID_PREVIEW); |  | ||||||
|             harbor().remove(selectedInHarbor); |  | ||||||
|             harbor().add(selectedInHarbor); |  | ||||||
|             preview = new Battleship(selectedInHarbor.getLength(), 0, 0, RIGHT); |  | ||||||
|             setPreviewStatus(preview); |  | ||||||
|             ownMap().add(preview); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Rotates the preview ship. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void rotateShip() { |  | ||||||
|         ClientGameLogic.LOGGER.log(Level.DEBUG, "pushed rotate"); //NON-NLS |  | ||||||
|         if (preview == null) return; |  | ||||||
|         preview.rotated(); |  | ||||||
|         ownMap().remove(preview); |  | ||||||
|         ownMap().add(preview); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Finds a free position in the harbor to place a ship. |  | ||||||
|      * |  | ||||||
|      * @return the y coordinate of a free position in the harbor |  | ||||||
|      */ |  | ||||||
|     private int freeY() { |  | ||||||
|         for (int i = 0; i < harbor().getHeight(); i++) |  | ||||||
|             if (harbor().findShipAt(0, i) == null) |  | ||||||
|                 return i; |  | ||||||
|         throw new RuntimeException("Cannot find a free slot in harbor"); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Updates the status of the specified ship based on its validity. |  | ||||||
|      */ |  | ||||||
|     private void setPreviewStatus(Battleship ship) { |  | ||||||
|         ship.setStatus(ownMap().isValid(ship) ? VALID_PREVIEW : INVALID_PREVIEW); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The user has marked the map as finished. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void mapFinished() { |  | ||||||
|         if (!harbor().getItems().isEmpty()) return; |  | ||||||
|         logic.send(new MapMessage(ownMap().getRemainingShips())); |  | ||||||
|         logic.setInfoText("wait.for.opponent"); |  | ||||||
|         logic.setState(new WaitState(logic)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the player's map is complete (i.e., all ships are placed). |  | ||||||
|      * |  | ||||||
|      * @return true if all ships are placed, false otherwise |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public boolean isMapComplete() { |  | ||||||
|         return harbor().getItems().isEmpty(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if there is currently a preview ship. |  | ||||||
|      * |  | ||||||
|      * @return true if there is currently a preview ship, false otherwise |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public boolean movingShip() { |  | ||||||
|         return preview != null; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the player's own map. |  | ||||||
|      * |  | ||||||
|      * @return the player's own map |  | ||||||
|      */ |  | ||||||
|     private ShipMap ownMap() { |  | ||||||
|         return logic.getOwnMap(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the harbor map. |  | ||||||
|      * |  | ||||||
|      * @return the harbor map |  | ||||||
|      */ |  | ||||||
|     private ShipMap harbor() { |  | ||||||
|         return logic.getHarbor(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Loads a map from the specified file. |  | ||||||
|      * |  | ||||||
|      * @param file the file to load the map from |  | ||||||
|      * @throws IOException if the map cannot be loaded |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void loadMap(File file) throws IOException { |  | ||||||
|         final ShipMapDTO dto = ShipMapDTO.loadFrom(file); |  | ||||||
|         if (!dto.fits(logic.getDetails())) |  | ||||||
|             throw new IOException(lookup("map.doesnt.fit")); |  | ||||||
|         ownMap().clear(); |  | ||||||
|         dto.getShips().forEach(ownMap()::add); |  | ||||||
|         harbor().clear(); |  | ||||||
|         preview = null; |  | ||||||
|         selectedInHarbor = null; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the player's own map may be loaded from a file. |  | ||||||
|      * |  | ||||||
|      * @return true if the own map may be loaded from file, false otherwise |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public boolean mayLoadMap() { |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the player's own map may be saved to a file. |  | ||||||
|      * |  | ||||||
|      * @return true if the own map may be saved to file, false otherwise |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public boolean maySaveMap() { |  | ||||||
|         return harbor().getItems().isEmpty(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,32 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.game.client; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Represents the state of the client when the game is over. |  | ||||||
|  */ |  | ||||||
| class GameOverState extends ClientState { |  | ||||||
|     /** |  | ||||||
|      * Constructs a new instance of GameOverState. |  | ||||||
|      * |  | ||||||
|      * @param logic the client game logic |  | ||||||
|      */ |  | ||||||
|     GameOverState(ClientGameLogic logic) { |  | ||||||
|         super(logic); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns true to indicate that the battle state should be shown. |  | ||||||
|      * |  | ||||||
|      * @return true if the battle state should be shown |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public boolean showBattle() { |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,65 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.game.client; |  | ||||||
|  |  | ||||||
| import pp.battleship.message.server.GameDetails; |  | ||||||
| import pp.battleship.model.Battleship; |  | ||||||
| import pp.battleship.model.Rotation; |  | ||||||
|  |  | ||||||
| import java.util.Map.Entry; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Represents the state of the client waiting for the |  | ||||||
|  * {@linkplain pp.battleship.message.server.GameDetails} |  | ||||||
|  * from the server. |  | ||||||
|  */ |  | ||||||
| class InitialState extends ClientState { |  | ||||||
|     /** |  | ||||||
|      * Creates a new initial state. |  | ||||||
|      * |  | ||||||
|      * @param logic the game logic |  | ||||||
|      */ |  | ||||||
|     public InitialState(ClientGameLogic logic) { |  | ||||||
|         super(logic); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Sets the game details provided by the server. |  | ||||||
|      * |  | ||||||
|      * @param details the game details including map size and ships |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void receivedGameDetails(GameDetails details) { |  | ||||||
|         logic.initializeMaps(details); |  | ||||||
|         fillHarbor(details); |  | ||||||
|         logic.setInfoText(details.getInfoTextKey()); |  | ||||||
|         logic.setState(new EditorState(logic)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Fills the harbor with ships as specified by the game details. |  | ||||||
|      * |  | ||||||
|      * @param details the game details including map size and ships |  | ||||||
|      */ |  | ||||||
|     private void fillHarbor(GameDetails details) { |  | ||||||
|         int y = 0; |  | ||||||
|         for (Entry<Integer, Integer> entry : details.getShipNums().entrySet()) { |  | ||||||
|             final int len = entry.getKey(); |  | ||||||
|             final int num = entry.getValue(); |  | ||||||
|             for (int i = 0; i < num; i++) { |  | ||||||
|                 final Battleship ship = new Battleship(len, 0, y++, Rotation.RIGHT); |  | ||||||
|                 logic.getHarbor().add(ship); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public boolean maySaveMap() { |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,32 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.game.client; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Interface representing a connection to the server. |  | ||||||
|  * Extends ClientSender to allow sending messages to the server. |  | ||||||
|  */ |  | ||||||
| public interface ServerConnection extends ClientSender { |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the client is currently connected to the server. |  | ||||||
|      * |  | ||||||
|      * @return true if connected, false otherwise. |  | ||||||
|      */ |  | ||||||
|     boolean isConnected(); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Establishes a connection to the server. |  | ||||||
|      */ |  | ||||||
|     void connect(); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Disconnects from the server. |  | ||||||
|      */ |  | ||||||
|     void disconnect(); |  | ||||||
| } |  | ||||||
| @@ -1,41 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.game.client; |  | ||||||
|  |  | ||||||
| import pp.battleship.message.server.StartBattleMessage; |  | ||||||
|  |  | ||||||
| import java.lang.System.Logger.Level; |  | ||||||
|  |  | ||||||
| class WaitState extends ClientState { |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates a new instance of {@link WaitState}. |  | ||||||
|      * |  | ||||||
|      * @param logic the game logic |  | ||||||
|      */ |  | ||||||
|     public WaitState(ClientGameLogic logic) { |  | ||||||
|         super(logic); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public boolean showEditor() { |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Starts the battle based on the server message. |  | ||||||
|      * |  | ||||||
|      * @param msg the message indicating whose turn it is to shoot |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void receivedStartBattle(StartBattleMessage msg) { |  | ||||||
|         ClientGameLogic.LOGGER.log(Level.INFO, "start battle, {0} turn", msg.isMyTurn() ? "my" : "other's"); //NON-NLS |  | ||||||
|         logic.setInfoText(msg.getInfoTextKey()); |  | ||||||
|         logic.setState(new BattleState(logic, msg.isMyTurn())); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,54 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.game.server; |  | ||||||
|  |  | ||||||
| import pp.battleship.BattleshipConfig; |  | ||||||
| import pp.battleship.model.ShipMap; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Class representing a player |  | ||||||
|  */ |  | ||||||
| public class Player { |  | ||||||
|     private final ShipMap map; |  | ||||||
|     private final String name; |  | ||||||
|     private final int id; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates new Player |  | ||||||
|      * |  | ||||||
|      * @param id     the id of the connection to the client represented by this player |  | ||||||
|      * @param name   the human-readable name of this player |  | ||||||
|      * @param config model holding the player |  | ||||||
|      */ |  | ||||||
|     Player(int id, String name, BattleshipConfig config) { |  | ||||||
|         this.id = id; |  | ||||||
|         this.name = name; |  | ||||||
|         map = new ShipMap(config.getMapWidth(), config.getMapHeight(), null); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the id of the connection to the client represented by this player. |  | ||||||
|      * |  | ||||||
|      * @return the id |  | ||||||
|      */ |  | ||||||
|     public int getId() { |  | ||||||
|         return id; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public String toString() { |  | ||||||
|         return String.format("Player(%s,%s)", name, id); //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * @return map containing own ships and shots |  | ||||||
|      */ |  | ||||||
|     public ShipMap getMap() { |  | ||||||
|         return map; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,220 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.game.server; |  | ||||||
|  |  | ||||||
| import pp.battleship.BattleshipConfig; |  | ||||||
| import pp.battleship.message.client.ClientInterpreter; |  | ||||||
| import pp.battleship.message.client.MapMessage; |  | ||||||
| import pp.battleship.message.client.ShootMessage; |  | ||||||
| import pp.battleship.message.server.EffectMessage; |  | ||||||
| import pp.battleship.message.server.GameDetails; |  | ||||||
| import pp.battleship.message.server.ServerMessage; |  | ||||||
| import pp.battleship.message.server.StartBattleMessage; |  | ||||||
| import pp.battleship.model.Battleship; |  | ||||||
| import pp.battleship.model.IntPoint; |  | ||||||
|  |  | ||||||
| import java.lang.System.Logger; |  | ||||||
| import java.lang.System.Logger.Level; |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.HashSet; |  | ||||||
| import java.util.List; |  | ||||||
| import java.util.Set; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Controls the server-side game logic for Battleship. |  | ||||||
|  * Manages game states, player interactions, and message handling. |  | ||||||
|  */ |  | ||||||
| public class ServerGameLogic implements ClientInterpreter { |  | ||||||
|     private static final Logger LOGGER = System.getLogger(ServerGameLogic.class.getName()); |  | ||||||
|  |  | ||||||
|     private final BattleshipConfig config; |  | ||||||
|     private final List<Player> players = new ArrayList<>(2); |  | ||||||
|     private final Set<Player> readyPlayers = new HashSet<>(); |  | ||||||
|     private final ServerSender serverSender; |  | ||||||
|     private Player activePlayer; |  | ||||||
|     private ServerState state = ServerState.WAIT; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a ServerGameLogic with the specified sender and configuration. |  | ||||||
|      * |  | ||||||
|      * @param serverSender the sender used to send messages to clients |  | ||||||
|      * @param config       the game configuration |  | ||||||
|      */ |  | ||||||
|     public ServerGameLogic(ServerSender serverSender, BattleshipConfig config) { |  | ||||||
|         this.serverSender = serverSender; |  | ||||||
|         this.config = config; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the  state of the game. |  | ||||||
|      */ |  | ||||||
|     ServerState getState() { |  | ||||||
|         return state; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Sets the new state of the game and logs the state transition. |  | ||||||
|      * |  | ||||||
|      * @param newState the new state to set |  | ||||||
|      */ |  | ||||||
|     void setState(ServerState newState) { |  | ||||||
|         LOGGER.log(Level.DEBUG, "state transition {0} --> {1}", state, newState); //NON-NLS |  | ||||||
|         state = newState; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the opponent of the specified player. |  | ||||||
|      * |  | ||||||
|      * @param p the player |  | ||||||
|      * @return the opponent of the player |  | ||||||
|      */ |  | ||||||
|     Player getOpponent(Player p) { |  | ||||||
|         if (players.size() != 2) |  | ||||||
|             throw new RuntimeException("trying to find opponent without having 2 players"); |  | ||||||
|         final int index = players.indexOf(p); |  | ||||||
|         if (index < 0) |  | ||||||
|             throw new RuntimeException("Nonexistent player " + p); |  | ||||||
|         return players.get(1 - index); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the player representing the client with the specified connection ID. |  | ||||||
|      * |  | ||||||
|      * @param id the ID of the client |  | ||||||
|      * @return the player associated with the client ID, or null if not found |  | ||||||
|      */ |  | ||||||
|     public Player getPlayerById(int id) { |  | ||||||
|         for (Player player : players) |  | ||||||
|             if (player.getId() == id) |  | ||||||
|                 return player; |  | ||||||
|         LOGGER.log(Level.ERROR, "no player found with connection {0}", id); //NON-NLS |  | ||||||
|         return null; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Sends a message to the specified player. |  | ||||||
|      * |  | ||||||
|      * @param player the player to send the message to |  | ||||||
|      * @param msg    the message to send |  | ||||||
|      */ |  | ||||||
|     void send(Player player, ServerMessage msg) { |  | ||||||
|         LOGGER.log(Level.INFO, "sending to {0}: {1}", player, msg); //NON-NLS |  | ||||||
|         serverSender.send(player.getId(), msg); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Adds a new player to the game if there are less than two players. |  | ||||||
|      * Transitions the state to SET_UP if two players are present. |  | ||||||
|      * |  | ||||||
|      * @param id the connection ID of the new player |  | ||||||
|      * @return the player added to the game, or null if the game is not in the right state |  | ||||||
|      */ |  | ||||||
|     public Player addPlayer(int id) { |  | ||||||
|         if (state != ServerState.WAIT) { |  | ||||||
|             LOGGER.log(Level.ERROR, "addPlayer not allowed in {0}", state); //NON-NLS |  | ||||||
|             return null; |  | ||||||
|         } |  | ||||||
|         final int n = players.size() + 1; |  | ||||||
|         final Player player = new Player(id, "player " + n, config); //NON-NLS |  | ||||||
|         LOGGER.log(Level.INFO, "adding {0}", player); //NON-NLS |  | ||||||
|         players.add(player); |  | ||||||
|         if (players.size() == 2) { |  | ||||||
|             activePlayer = players.get(0); |  | ||||||
|             for (Player p : players) |  | ||||||
|                 send(p, new GameDetails(config)); |  | ||||||
|             setState(ServerState.SET_UP); |  | ||||||
|         } |  | ||||||
|         return player; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles the reception of a MapMessage. |  | ||||||
|      * |  | ||||||
|      * @param msg  the received MapMessage |  | ||||||
|      * @param from the ID of the sender client |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void received(MapMessage msg, int from) { |  | ||||||
|         if (state != ServerState.SET_UP) |  | ||||||
|             LOGGER.log(Level.ERROR, "playerReady not allowed in {0}", state); //NON-NLS |  | ||||||
|         else |  | ||||||
|             playerReady(getPlayerById(from), msg.getShips()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles the reception of a ShootMessage. |  | ||||||
|      * |  | ||||||
|      * @param msg  the received ShootMessage |  | ||||||
|      * @param from the ID of the sender client |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void received(ShootMessage msg, int from) { |  | ||||||
|         if (state != ServerState.BATTLE) |  | ||||||
|             LOGGER.log(Level.ERROR, "shoot not allowed in {0}", state); //NON-NLS |  | ||||||
|         else |  | ||||||
|             shoot(getPlayerById(from), msg.getPosition()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Marks the player as ready and sets their ships. |  | ||||||
|      * Transitions the state to PLAY if both players are ready. |  | ||||||
|      * |  | ||||||
|      * @param player the player who is ready |  | ||||||
|      * @param ships  the list of ships placed by the player |  | ||||||
|      */ |  | ||||||
|     void playerReady(Player player, List<Battleship> ships) { |  | ||||||
|         if (!readyPlayers.add(player)) { |  | ||||||
|             LOGGER.log(Level.ERROR, "{0} was already ready", player); //NON-NLS |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         ships.forEach(player.getMap()::add); |  | ||||||
|         if (readyPlayers.size() == 2) { |  | ||||||
|             for (Player p : players) |  | ||||||
|                 send(p, new StartBattleMessage(p == activePlayer)); |  | ||||||
|             setState(ServerState.BATTLE); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles the shooting action by the player. |  | ||||||
|      * |  | ||||||
|      * @param p   the player who shot |  | ||||||
|      * @param pos the position of the shot |  | ||||||
|      */ |  | ||||||
|     void shoot(Player p, IntPoint pos) { |  | ||||||
|         if (p != activePlayer) return; |  | ||||||
|         final Player otherPlayer = getOpponent(activePlayer); |  | ||||||
|         final Battleship selectedShip = otherPlayer.getMap().findShipAt(pos); |  | ||||||
|         if (selectedShip == null) { |  | ||||||
|             // shot missed |  | ||||||
|             send(activePlayer, EffectMessage.miss(true, pos)); |  | ||||||
|             send(otherPlayer, EffectMessage.miss(false, pos)); |  | ||||||
|             activePlayer = otherPlayer; |  | ||||||
|         } |  | ||||||
|         else { |  | ||||||
|             // shot hit a ship |  | ||||||
|             selectedShip.hit(pos); |  | ||||||
|             if (otherPlayer.getMap().getRemainingShips().isEmpty()) { |  | ||||||
|                 // game is over |  | ||||||
|                 send(activePlayer, EffectMessage.won(pos, selectedShip)); |  | ||||||
|                 send(otherPlayer, EffectMessage.lost(pos, selectedShip, activePlayer.getMap().getRemainingShips())); |  | ||||||
|                 setState(ServerState.GAME_OVER); |  | ||||||
|             } |  | ||||||
|             else if (selectedShip.isDestroyed()) { |  | ||||||
|                 // ship has been destroyed, but game is not yet over |  | ||||||
|                 send(activePlayer, EffectMessage.shipDestroyed(true, pos, selectedShip)); |  | ||||||
|                 send(otherPlayer, EffectMessage.shipDestroyed(false, pos, selectedShip)); |  | ||||||
|             } |  | ||||||
|             else { |  | ||||||
|                 // ship has been hit, but it hasn't been destroyed |  | ||||||
|                 send(activePlayer, EffectMessage.hit(true, pos)); |  | ||||||
|                 send(otherPlayer, EffectMessage.hit(false, pos)); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,23 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.game.server; |  | ||||||
|  |  | ||||||
| import pp.battleship.message.server.ServerMessage; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Interface for sending messages to a client. |  | ||||||
|  */ |  | ||||||
| public interface ServerSender { |  | ||||||
|     /** |  | ||||||
|      * Send the specified message to the client. |  | ||||||
|      * |  | ||||||
|      * @param id      the id of the client that shall receive the message |  | ||||||
|      * @param message the message |  | ||||||
|      */ |  | ||||||
|     void send(int id, ServerMessage message); |  | ||||||
| } |  | ||||||
| @@ -1,33 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.game.server; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Represents the different states of the Battleship server during the game lifecycle. |  | ||||||
|  */ |  | ||||||
| enum ServerState { |  | ||||||
|     /** |  | ||||||
|      * The server is waiting for clients to connect. |  | ||||||
|      */ |  | ||||||
|     WAIT, |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The server is waiting for clients to set up their maps. |  | ||||||
|      */ |  | ||||||
|     SET_UP, |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The battle of the game where players take turns to attack each other's ships. |  | ||||||
|      */ |  | ||||||
|     BATTLE, |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The game has ended because all the ships of one player have been destroyed. |  | ||||||
|      */ |  | ||||||
|     GAME_OVER |  | ||||||
| } |  | ||||||
| @@ -1,114 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.game.singlemode; |  | ||||||
|  |  | ||||||
| import pp.battleship.BattleshipConfig; |  | ||||||
| import pp.battleship.model.IntPoint; |  | ||||||
|  |  | ||||||
| import java.io.File; |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.Iterator; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Class providing access to the Battleship client configuration. |  | ||||||
|  * Extends {@link BattleshipConfig} to include additional properties specific to the client. |  | ||||||
|  * This class manages configuration settings related to the RobotClient's behavior |  | ||||||
|  * and the game maps used in single mode. |  | ||||||
|  * <p> |  | ||||||
|  * <b>Note:</b> Attributes of this class should not be marked as {@code final} |  | ||||||
|  * to ensure proper functionality when reading from a properties file. |  | ||||||
|  * </p> |  | ||||||
|  */ |  | ||||||
| public class BattleshipClientConfig extends BattleshipConfig { |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Array representing the predefined shooting locations for the RobotClient. |  | ||||||
|      * The array stores coordinates in pairs, where even indices represent x-coordinates |  | ||||||
|      * and odd indices represent y-coordinates. |  | ||||||
|      */ |  | ||||||
|     @Property("robot.targets") |  | ||||||
|     private int[] robotTargets = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9}; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The delay (in milliseconds) between shots fired by the RobotClient. |  | ||||||
|      */ |  | ||||||
|     @Property("robot.delay") |  | ||||||
|     private int delay = 750; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Path to the file representing the opponent's map. |  | ||||||
|      */ |  | ||||||
|     @Property("map.opponent") |  | ||||||
|     private String opponentMap; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Path to the file representing the player's own map. |  | ||||||
|      */ |  | ||||||
|     @Property("map.own") |  | ||||||
|     private String ownMap; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates a default {@code BattleshipClientConfig} with predefined values. |  | ||||||
|      */ |  | ||||||
|     public BattleshipClientConfig() { |  | ||||||
|         // Default constructor |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns an iterator of {@link IntPoint} objects representing the predefined |  | ||||||
|      * shooting locations for the RobotClient. |  | ||||||
|      * |  | ||||||
|      * @return an iterator of {@code IntPoint} representing the shooting locations. |  | ||||||
|      */ |  | ||||||
|     public Iterator<IntPoint> getRobotTargets() { |  | ||||||
|         List<IntPoint> targets = new ArrayList<>(); |  | ||||||
|         for (int i = 0; i < robotTargets.length; i += 2) { |  | ||||||
|             int x = robotTargets[i]; |  | ||||||
|             int y = robotTargets[i + 1]; |  | ||||||
|             targets.add(new IntPoint(x, y)); |  | ||||||
|         } |  | ||||||
|         return targets.iterator(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the delay (in milliseconds) between shots by the RobotClient. |  | ||||||
|      * |  | ||||||
|      * @return the delay in milliseconds. |  | ||||||
|      */ |  | ||||||
|     public int getDelay() { |  | ||||||
|         return delay; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the file representing the opponent's map. |  | ||||||
|      * |  | ||||||
|      * @return the opponent's map file, or {@code null} if not set. |  | ||||||
|      */ |  | ||||||
|     public File getOpponentMap() { |  | ||||||
|         return opponentMap == null ? null : new File(opponentMap); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the file representing the player's own map. |  | ||||||
|      * |  | ||||||
|      * @return the player's own map file, or {@code null} if not set. |  | ||||||
|      */ |  | ||||||
|     public File getOwnMap() { |  | ||||||
|         return ownMap == null ? null : new File(ownMap); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Determines if the game is in single mode based on the presence of an opponent map. |  | ||||||
|      * |  | ||||||
|      * @return {@code true} if the opponent map is set, indicating single mode; {@code false} otherwise. |  | ||||||
|      */ |  | ||||||
|     public boolean isSingleMode() { |  | ||||||
|         return opponentMap != null; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,75 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.game.singlemode; |  | ||||||
|  |  | ||||||
| import pp.battleship.message.client.ClientInterpreter; |  | ||||||
| import pp.battleship.message.client.ClientMessage; |  | ||||||
| import pp.battleship.message.client.MapMessage; |  | ||||||
| import pp.battleship.message.client.ShootMessage; |  | ||||||
| import pp.battleship.model.Battleship; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * The {@code Copycat} class is a utility that creates a copy of a {@link ClientMessage}. |  | ||||||
|  * It implements the {@link ClientInterpreter} interface to interpret and process |  | ||||||
|  * different types of messages. |  | ||||||
|  */ |  | ||||||
| class Copycat implements ClientInterpreter { |  | ||||||
|     private ClientMessage copiedMessage; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates a copy of the provided {@link ClientMessage}. |  | ||||||
|      * |  | ||||||
|      * @param msg the message to be copied |  | ||||||
|      * @return a copy of the provided message |  | ||||||
|      */ |  | ||||||
|     static ClientMessage copy(ClientMessage msg) { |  | ||||||
|         final Copycat copycat = new Copycat(); |  | ||||||
|         msg.accept(copycat, 0); |  | ||||||
|         return copycat.copiedMessage; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Private constructor to prevent direct instantiation of {@code Copycat}. |  | ||||||
|      */ |  | ||||||
|     private Copycat() { /* do nothing */ } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles the reception of a {@link ShootMessage}. |  | ||||||
|      * Since a {@code ShootMessage} does not need to be copied, it is directly assigned. |  | ||||||
|      * |  | ||||||
|      * @param msg  the received {@code ShootMessage} |  | ||||||
|      * @param from the identifier of the sender |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void received(ShootMessage msg, int from) { |  | ||||||
|         // copying is not necessary |  | ||||||
|         copiedMessage = msg; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles the reception of a {@link MapMessage}. |  | ||||||
|      * Creates a deep copy of the {@code MapMessage} by copying each {@link Battleship} in the message. |  | ||||||
|      * |  | ||||||
|      * @param msg  the received {@code MapMessage} |  | ||||||
|      * @param from the identifier of the sender |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void received(MapMessage msg, int from) { |  | ||||||
|         copiedMessage = new MapMessage(msg.getShips().stream().map(Copycat::copy).toList()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates a copy of the provided {@link Battleship}. |  | ||||||
|      * |  | ||||||
|      * @param ship the battleship to be copied |  | ||||||
|      * @return a copy of the provided battleship |  | ||||||
|      */ |  | ||||||
|     private static Battleship copy(Battleship ship) { |  | ||||||
|         return new Battleship(ship.getLength(), ship.getX(), ship.getY(), ship.getRot()); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,93 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.game.singlemode; |  | ||||||
|  |  | ||||||
| import pp.battleship.game.client.BattleshipClient; |  | ||||||
| import pp.battleship.game.client.ClientGameLogic; |  | ||||||
| import pp.battleship.message.server.EffectMessage; |  | ||||||
| import pp.battleship.message.server.GameDetails; |  | ||||||
| import pp.battleship.message.server.ServerInterpreter; |  | ||||||
| import pp.battleship.message.server.ServerMessage; |  | ||||||
| import pp.battleship.message.server.StartBattleMessage; |  | ||||||
|  |  | ||||||
| import java.io.IOException; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * A proxy class that interprets messages from the server and forwards them to the BattleshipClient. |  | ||||||
|  * Implements the ServerInterpreter interface to handle specific server messages. |  | ||||||
|  */ |  | ||||||
| class InterpreterProxy implements ServerInterpreter { |  | ||||||
|     private final BattleshipClient playerClient; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs an InterpreterProxy with the specified BattleshipClient. |  | ||||||
|      * |  | ||||||
|      * @param playerClient the client to which the server messages are forwarded |  | ||||||
|      */ |  | ||||||
|     InterpreterProxy(BattleshipClient playerClient) { |  | ||||||
|         this.playerClient = playerClient; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles the received GameDetails message by accepting it with the client's game logic. |  | ||||||
|      * If the client's own map option is set, it also loads the map. |  | ||||||
|      * |  | ||||||
|      * @param msg the GameDetails message received from the server |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void received(GameDetails msg) { |  | ||||||
|         msg.accept(playerClient.getGameLogic()); |  | ||||||
|         if (playerClient.getConfig().getOwnMap() != null) |  | ||||||
|             playerClient.enqueue(this::loadMap); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Loads the map specified in the client's options and notifies the game logic that the map is finished. |  | ||||||
|      * |  | ||||||
|      * @throws RuntimeException if the map fails to load. |  | ||||||
|      */ |  | ||||||
|     private void loadMap() { |  | ||||||
|         final ClientGameLogic clientGameLogic = playerClient.getGameLogic(); |  | ||||||
|         try { |  | ||||||
|             clientGameLogic.loadMap(playerClient.getConfig().getOwnMap()); |  | ||||||
|         } |  | ||||||
|         catch (IOException e) { |  | ||||||
|             throw new RuntimeException("Failed to load PlayerClient map", e); |  | ||||||
|         } |  | ||||||
|         clientGameLogic.mapFinished(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Forwards the received StartBattleMessage to the client's game logic. |  | ||||||
|      * |  | ||||||
|      * @param msg the StartBattleMessage received from the server |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void received(StartBattleMessage msg) { |  | ||||||
|         forward(msg); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Forwards the received EffectMessage to the client's game logic. |  | ||||||
|      * |  | ||||||
|      * @param msg the EffectMessage received from the server |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void received(EffectMessage msg) { |  | ||||||
|         forward(msg); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Forwards the specified ServerMessage to the client's game logic by enqueuing the message acceptance. |  | ||||||
|      * |  | ||||||
|      * @param msg the ServerMessage to forward |  | ||||||
|      */ |  | ||||||
|     private void forward(ServerMessage msg) { |  | ||||||
|         playerClient.enqueue(() -> msg.accept(playerClient.getGameLogic())); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,127 +0,0 @@ | |||||||
| package pp.battleship.game.singlemode; |  | ||||||
|  |  | ||||||
| import pp.battleship.game.client.BattleshipClient; |  | ||||||
| import pp.battleship.message.client.MapMessage; |  | ||||||
| import pp.battleship.message.client.ShootMessage; |  | ||||||
| import pp.battleship.message.server.EffectMessage; |  | ||||||
| import pp.battleship.message.server.GameDetails; |  | ||||||
| import pp.battleship.message.server.ServerInterpreter; |  | ||||||
| import pp.battleship.message.server.StartBattleMessage; |  | ||||||
| import pp.battleship.model.IntPoint; |  | ||||||
| import pp.battleship.model.dto.ShipMapDTO; |  | ||||||
| import pp.util.RandomPositionIterator; |  | ||||||
|  |  | ||||||
| import java.lang.System.Logger; |  | ||||||
| import java.lang.System.Logger.Level; |  | ||||||
| import java.util.HashSet; |  | ||||||
| import java.util.Iterator; |  | ||||||
| import java.util.Set; |  | ||||||
| import java.util.Timer; |  | ||||||
| import java.util.TimerTask; |  | ||||||
|  |  | ||||||
| import static pp.battleship.Resources.lookup; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * RobotClient simulates a client in the Battleship game. |  | ||||||
|  * It handles its own shooting targets and acts as a sender of client messages. |  | ||||||
|  * The RobotClient can shoot at predefined targets or generate random targets if predefined ones are exhausted. |  | ||||||
|  */ |  | ||||||
| class RobotClient implements ServerInterpreter { |  | ||||||
|     private static final Logger LOGGER = System.getLogger(RobotClient.class.getName()); |  | ||||||
|     private final BattleshipClient app; |  | ||||||
|     private final ServerConnectionMockup connection; |  | ||||||
|     private final BattleshipClientConfig config; |  | ||||||
|     private final ShipMapDTO dto; |  | ||||||
|     private final Iterator<IntPoint> targetIterator; |  | ||||||
|     private final Iterator<IntPoint> randomPositionIterator; |  | ||||||
|     private final Set<IntPoint> shotTargets = new HashSet<>(); |  | ||||||
|     private final Timer timer = new Timer(true); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a RobotClient instance with the given connection and configuration. |  | ||||||
|      * Initializes the shooting targets from the configuration. |  | ||||||
|      * |  | ||||||
|      * @param app        The BattleshipApp instance, used to enqueue actions on the main thread |  | ||||||
|      * @param connection The ServerConnectionMockup instance |  | ||||||
|      * @param config     The BattleshipClientConfig instance |  | ||||||
|      * @param dto        The ShipMap dto specified at app start |  | ||||||
|      */ |  | ||||||
|     RobotClient(BattleshipClient app, ServerConnectionMockup connection, BattleshipClientConfig config, ShipMapDTO dto) { |  | ||||||
|         this.app = app; |  | ||||||
|         this.connection = connection; |  | ||||||
|         this.config = config; |  | ||||||
|         this.dto = dto; |  | ||||||
|         this.targetIterator = config.getRobotTargets(); |  | ||||||
|         this.randomPositionIterator = new RandomPositionIterator<>(IntPoint::new, config.getMapWidth(), config.getMapHeight()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Schedules the RobotClient to take a shot after the specified delay. |  | ||||||
|      */ |  | ||||||
|     private void shoot() { |  | ||||||
|         timer.schedule(new TimerTask() { |  | ||||||
|             @Override |  | ||||||
|             public void run() { |  | ||||||
|                 app.enqueue(RobotClient.this::robotShot); |  | ||||||
|             } |  | ||||||
|         }, config.getDelay()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Makes the RobotClient take a shot by sending a ShootMessage with the target position. |  | ||||||
|      */ |  | ||||||
|     private void robotShot() { |  | ||||||
|         connection.sendRobotMessage(new ShootMessage(getShotPosition())); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Determines the next shot position. If predefined targets are available, uses the next target. |  | ||||||
|      * Otherwise, generates a random target that has not been shot at before. |  | ||||||
|      * |  | ||||||
|      * @return the next shot position as IntPosition |  | ||||||
|      */ |  | ||||||
|     private IntPoint getShotPosition() { |  | ||||||
|         while (true) { |  | ||||||
|             final IntPoint target = targetIterator.hasNext() ? targetIterator.next() : randomPositionIterator.next(); |  | ||||||
|             if (shotTargets.add(target)) return target; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Receives GameDetails, creates and sends the ShipMap to the mock server. |  | ||||||
|      * |  | ||||||
|      * @param details The game details |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void received(GameDetails details) { |  | ||||||
|         if (!dto.fits(details)) |  | ||||||
|             throw new RuntimeException(lookup("map.doesnt.fit")); |  | ||||||
|         app.enqueue(() -> connection.sendRobotMessage(new MapMessage(dto.getShips()))); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Receives the StartBattleMessage and updates the turn status. |  | ||||||
|      * If it is RobotClient's turn to shoot, schedules a shot using shoot(); |  | ||||||
|      * |  | ||||||
|      * @param msg The start battle message |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void received(StartBattleMessage msg) { |  | ||||||
|         LOGGER.log(Level.INFO, "Received StartBattleMessage: {0}", msg); //NON-NLS |  | ||||||
|         if (msg.isMyTurn()) |  | ||||||
|             shoot(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Receives an effect message, logs it, and updates the turn status. |  | ||||||
|      * If it is RobotClient's turn to shoot, schedules a shot using shoot(); |  | ||||||
|      * |  | ||||||
|      * @param msg The effect message |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void received(EffectMessage msg) { |  | ||||||
|         LOGGER.log(Level.INFO, "Received EffectMessage: {0}", msg); //NON-NLS |  | ||||||
|         if (msg.isMyTurn()) |  | ||||||
|             shoot(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,145 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.game.singlemode; |  | ||||||
|  |  | ||||||
| import pp.battleship.game.client.BattleshipClient; |  | ||||||
| import pp.battleship.game.client.ServerConnection; |  | ||||||
| import pp.battleship.game.server.ServerGameLogic; |  | ||||||
| import pp.battleship.game.server.ServerSender; |  | ||||||
| import pp.battleship.message.client.ClientMessage; |  | ||||||
| import pp.battleship.message.server.ServerInterpreter; |  | ||||||
| import pp.battleship.message.server.ServerMessage; |  | ||||||
| import pp.battleship.model.dto.ShipMapDTO; |  | ||||||
|  |  | ||||||
| import java.io.IOException; |  | ||||||
|  |  | ||||||
| import static pp.battleship.game.singlemode.Copycat.copy; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * A mock implementation of the ServerConnection interface for single mode. |  | ||||||
|  * Simulates a server connection without actual network communication. |  | ||||||
|  * Handles a mock player named RobotClient and schedules its shots when due. |  | ||||||
|  */ |  | ||||||
| public class ServerConnectionMockup implements ServerConnection, ServerSender { |  | ||||||
|     /** |  | ||||||
|      * The id of the player client. |  | ||||||
|      */ |  | ||||||
|     private static final int PLAYER_CLIENT = 1; |  | ||||||
|     /** |  | ||||||
|      * The id of the robot client. |  | ||||||
|      */ |  | ||||||
|     private static final int ROBOT_CLIENT = 2; |  | ||||||
|  |  | ||||||
|     private final BattleshipClient playerClient; |  | ||||||
|     private final RobotClient robotClient; |  | ||||||
|     private final InterpreterProxy playerProxy; |  | ||||||
|     private final ServerGameLogic serverGameLogic; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a ServerConnectionMockup instance for the given Battleship application. |  | ||||||
|      * Creates a RobotClient instance and an instance of ServerGameLogic. |  | ||||||
|      * |  | ||||||
|      * @param playerClient The Battleship client instance, e.g., a BattleshipApp instance. |  | ||||||
|      */ |  | ||||||
|     public ServerConnectionMockup(BattleshipClient playerClient) { |  | ||||||
|         this.playerClient = playerClient; |  | ||||||
|         robotClient = new RobotClient(playerClient, this, playerClient.getConfig(), getShipMapDTO()); |  | ||||||
|         serverGameLogic = new ServerGameLogic(this, playerClient.getConfig()); |  | ||||||
|         playerProxy = new InterpreterProxy(playerClient); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Always returns true as this is a mock connection. |  | ||||||
|      * |  | ||||||
|      * @return true, indicating the mock connection is always considered connected. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public boolean isConnected() { |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Simulates connecting to a server by adding the PlayerClient and the RobotClient to the serverGameLogic. |  | ||||||
|      * Loads the map of the PlayerClient and triggers sending it to the serverGameLogic. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void connect() { |  | ||||||
|         serverGameLogic.addPlayer(PLAYER_CLIENT); |  | ||||||
|         serverGameLogic.addPlayer(ROBOT_CLIENT); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Does nothing upon shutdown of the app. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void disconnect() { |  | ||||||
|         //do nothing |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Forwards the specified message received from the player client to the server logic. |  | ||||||
|      * |  | ||||||
|      * @param message The message from the player client to be processed. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void send(ClientMessage message) { // from PlayerClient, as this is the PlayerClients 'serverConnection' |  | ||||||
|         copy(message).accept(serverGameLogic, PLAYER_CLIENT); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Forwards the specified message received from the robot client to the server logic. |  | ||||||
|      * |  | ||||||
|      * @param message The message from the robot client to be processed. |  | ||||||
|      */ |  | ||||||
|     void sendRobotMessage(ClientMessage message) { |  | ||||||
|         message.accept(serverGameLogic, ROBOT_CLIENT); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Forwards the specified message received from the server logic either to the player client or to the |  | ||||||
|      * robot client, depending on the specified id. |  | ||||||
|      * |  | ||||||
|      * @param id      The recipient id |  | ||||||
|      * @param message The server message to be processed |  | ||||||
|      * @see #PLAYER_CLIENT |  | ||||||
|      * @see #ROBOT_CLIENT |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void send(int id, ServerMessage message) { |  | ||||||
|         message.accept(getInterpreter(id)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Retrieves the ServerInterpreter of the client with the specified id. |  | ||||||
|      * |  | ||||||
|      * @param clientId the id of the client whose ServerInterpreter shall be retrieved. |  | ||||||
|      * @return the ServerInterpreter of the client |  | ||||||
|      * @throws java.lang.IllegalArgumentException if there is no client with the specified id. |  | ||||||
|      */ |  | ||||||
|     private ServerInterpreter getInterpreter(int clientId) { |  | ||||||
|         return switch (clientId) { |  | ||||||
|             case PLAYER_CLIENT -> playerProxy; |  | ||||||
|             case ROBOT_CLIENT -> robotClient; |  | ||||||
|             default -> throw new IllegalArgumentException("Unexpected value: " + clientId); |  | ||||||
|         }; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Loads the ShipMapDTO from the opponent map file. |  | ||||||
|      * |  | ||||||
|      * @return the loaded ShipMapDTO. |  | ||||||
|      */ |  | ||||||
|     private ShipMapDTO getShipMapDTO() { |  | ||||||
|         try { |  | ||||||
|             return ShipMapDTO.loadFrom(playerClient.getConfig().getOpponentMap()); |  | ||||||
|         } |  | ||||||
|         catch (IOException e) { |  | ||||||
|             throw new RuntimeException("Failed to load RobotClient map", e); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,29 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.message.client; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Visitor interface for processing all client messages. |  | ||||||
|  */ |  | ||||||
| public interface ClientInterpreter { |  | ||||||
|     /** |  | ||||||
|      * Processes a received ShootMessage. |  | ||||||
|      * |  | ||||||
|      * @param msg  the ShootMessage to be processed |  | ||||||
|      * @param from the connection ID from which the message was received |  | ||||||
|      */ |  | ||||||
|     void received(ShootMessage msg, int from); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Processes a received MapMessage. |  | ||||||
|      * |  | ||||||
|      * @param msg  the MapMessage to be processed |  | ||||||
|      * @param from the connection ID from which the message was received |  | ||||||
|      */ |  | ||||||
|     void received(MapMessage msg, int from); |  | ||||||
| } |  | ||||||
| @@ -1,32 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.message.client; |  | ||||||
|  |  | ||||||
| import com.jme3.network.AbstractMessage; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * An abstract base class for client messages used in network transfer. |  | ||||||
|  * It extends the AbstractMessage class provided by the jme3-network library. |  | ||||||
|  */ |  | ||||||
| public abstract class ClientMessage extends AbstractMessage { |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a new ClientMessage instance. |  | ||||||
|      */ |  | ||||||
|     protected ClientMessage() { |  | ||||||
|         super(true); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Accepts a visitor for processing this message. |  | ||||||
|      * |  | ||||||
|      * @param interpreter the visitor to be used for processing |  | ||||||
|      * @param from        the connection ID of the sender |  | ||||||
|      */ |  | ||||||
|     public abstract void accept(ClientInterpreter interpreter, int from); |  | ||||||
| } |  | ||||||
| @@ -1,66 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.message.client; |  | ||||||
|  |  | ||||||
| import com.jme3.network.serializing.Serializable; |  | ||||||
| import pp.battleship.model.Battleship; |  | ||||||
|  |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * A message sent by the client containing the positions of the ships on the player's map. |  | ||||||
|  */ |  | ||||||
| @Serializable |  | ||||||
| public class MapMessage extends ClientMessage { |  | ||||||
|     private List<Battleship> ships; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Default constructor for serialization purposes. |  | ||||||
|      */ |  | ||||||
|     private MapMessage() { /* empty */ } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a MapMessage with the specified list of ships. |  | ||||||
|      * |  | ||||||
|      * @param ships the list of ships placed on the player's map |  | ||||||
|      */ |  | ||||||
|     public MapMessage(List<Battleship> ships) { |  | ||||||
|         this.ships = new ArrayList<>(ships); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the list of ships on the player's map. |  | ||||||
|      * |  | ||||||
|      * @return the list of ships |  | ||||||
|      */ |  | ||||||
|     public List<Battleship> getShips() { |  | ||||||
|         return ships; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns a string representation of the MapMessage. |  | ||||||
|      * |  | ||||||
|      * @return a string representation of the MapMessage |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public String toString() { |  | ||||||
|         return "MapMessage{ships=" + ships + '}'; //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Accepts a visitor to process this message. |  | ||||||
|      * |  | ||||||
|      * @param interpreter the visitor to process this message |  | ||||||
|      * @param from        the connection ID from which the message was received |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void accept(ClientInterpreter interpreter, int from) { |  | ||||||
|         interpreter.received(this, from); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,63 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.message.client; |  | ||||||
|  |  | ||||||
| import com.jme3.network.serializing.Serializable; |  | ||||||
| import pp.battleship.model.IntPoint; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * A message sent by the client to indicate a shooting action in the game. |  | ||||||
|  */ |  | ||||||
| @Serializable |  | ||||||
| public class ShootMessage extends ClientMessage { |  | ||||||
|     private IntPoint position; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Default constructor for serialization purposes. |  | ||||||
|      */ |  | ||||||
|     private ShootMessage() { /* empty */ } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a ShootMessage with the specified position. |  | ||||||
|      * |  | ||||||
|      * @param position the position where the shot is fired |  | ||||||
|      */ |  | ||||||
|     public ShootMessage(IntPoint position) { |  | ||||||
|         this.position = position; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the position of the shot. |  | ||||||
|      * |  | ||||||
|      * @return the position of the shot |  | ||||||
|      */ |  | ||||||
|     public IntPoint getPosition() { |  | ||||||
|         return position; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns a string representation of the ShootMessage. |  | ||||||
|      * |  | ||||||
|      * @return a string representation of the ShootMessage |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public String toString() { |  | ||||||
|         return "ShootMessage{position=" + position + '}'; //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Accepts a visitor to process this message. |  | ||||||
|      * |  | ||||||
|      * @param interpreter the visitor to process this message |  | ||||||
|      * @param from        the connection ID from which the message was received |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void accept(ClientInterpreter interpreter, int from) { |  | ||||||
|         interpreter.received(this, from); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,211 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.message.server; |  | ||||||
|  |  | ||||||
| import com.jme3.network.serializing.Serializable; |  | ||||||
| import pp.battleship.model.Battleship; |  | ||||||
| import pp.battleship.model.IntPoint; |  | ||||||
| import pp.battleship.model.Shot; |  | ||||||
|  |  | ||||||
| import java.util.Collections; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| import static pp.util.Util.copy; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * A message sent by the server to inform clients about the effects of a shot in the Battleship game. |  | ||||||
|  */ |  | ||||||
| @Serializable |  | ||||||
| public class EffectMessage extends ServerMessage { |  | ||||||
|     private boolean ownShot; |  | ||||||
|     private Shot shot; |  | ||||||
|     private Battleship destroyedShip; |  | ||||||
|     private List<Battleship> remainingOpponentShips; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates an EffectMessage indicating a hit. |  | ||||||
|      * |  | ||||||
|      * @param ownShot true if the shot was fired by the player, false if by the opponent |  | ||||||
|      * @param pos     the position of the shot |  | ||||||
|      * @return an EffectMessage indicating a hit |  | ||||||
|      */ |  | ||||||
|     public static EffectMessage hit(boolean ownShot, IntPoint pos) { |  | ||||||
|         return new EffectMessage(ownShot, new Shot(pos, true), null, null); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates an EffectMessage indicating a miss. |  | ||||||
|      * |  | ||||||
|      * @param ownShot true if the shot was fired by the player, false if by the opponent |  | ||||||
|      * @param pos     the position of the shot |  | ||||||
|      * @return an EffectMessage indicating a miss |  | ||||||
|      */ |  | ||||||
|     public static EffectMessage miss(boolean ownShot, IntPoint pos) { |  | ||||||
|         return new EffectMessage(ownShot, new Shot(pos, false), null, null); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates an EffectMessage indicating a ship was destroyed. |  | ||||||
|      * |  | ||||||
|      * @param ownShot       true if the shot was fired by the player, false if by the opponent |  | ||||||
|      * @param pos           the position of the shot |  | ||||||
|      * @param destroyedShip the ship that was destroyed |  | ||||||
|      * @return an EffectMessage indicating a ship was destroyed |  | ||||||
|      */ |  | ||||||
|     public static EffectMessage shipDestroyed(boolean ownShot, IntPoint pos, Battleship destroyedShip) { |  | ||||||
|         return new EffectMessage(ownShot, new Shot(pos, true), destroyedShip, null); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates an EffectMessage indicating the player has won the game. |  | ||||||
|      * |  | ||||||
|      * @param pos           the position of the shot |  | ||||||
|      * @param destroyedShip the ship that was destroyed |  | ||||||
|      * @return an EffectMessage indicating the player has won |  | ||||||
|      */ |  | ||||||
|     public static EffectMessage won(IntPoint pos, Battleship destroyedShip) { |  | ||||||
|         return new EffectMessage(true, new Shot(pos, true), destroyedShip, Collections.emptyList()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates an EffectMessage indicating the player has lost the game. |  | ||||||
|      * |  | ||||||
|      * @param pos                    the position of the shot |  | ||||||
|      * @param destroyedShip          the ship that was destroyed |  | ||||||
|      * @param remainingOpponentShips the list of opponent's remaining ships |  | ||||||
|      * @return an EffectMessage indicating the player has lost |  | ||||||
|      */ |  | ||||||
|     public static EffectMessage lost(IntPoint pos, Battleship destroyedShip, List<Battleship> remainingOpponentShips) { |  | ||||||
|         return new EffectMessage(false, new Shot(pos, true), destroyedShip, remainingOpponentShips); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Default constructor for serialization purposes. |  | ||||||
|      */ |  | ||||||
|     private EffectMessage() { /* empty */ } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs an EffectMessage with the specified parameters. |  | ||||||
|      * |  | ||||||
|      * @param ownShot                true if the shot was fired by the player, false if by the opponent |  | ||||||
|      * @param shot                   the shot fired |  | ||||||
|      * @param destroyedShip          the ship that was destroyed by the shot, null if no ship was destroyed |  | ||||||
|      * @param remainingOpponentShips the list of opponent's remaining ships after the shot |  | ||||||
|      */ |  | ||||||
|     private EffectMessage(boolean ownShot, Shot shot, Battleship destroyedShip, List<Battleship> remainingOpponentShips) { |  | ||||||
|         this.ownShot = ownShot; |  | ||||||
|         this.shot = shot; |  | ||||||
|         this.destroyedShip = destroyedShip; |  | ||||||
|         this.remainingOpponentShips = copy(remainingOpponentShips); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Accepts a visitor to process this message. |  | ||||||
|      * |  | ||||||
|      * @param interpreter the visitor to process this message |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void accept(ServerInterpreter interpreter) { |  | ||||||
|         interpreter.received(this); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the shot was fired by the player. |  | ||||||
|      * |  | ||||||
|      * @return true if the shot was fired by the player, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean isOwnShot() { |  | ||||||
|         return ownShot; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the shot fired. |  | ||||||
|      * |  | ||||||
|      * @return the shot fired |  | ||||||
|      */ |  | ||||||
|     public Shot getShot() { |  | ||||||
|         return shot; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the ship that was destroyed by the shot. |  | ||||||
|      * |  | ||||||
|      * @return the destroyed ship, null if no ship was destroyed |  | ||||||
|      */ |  | ||||||
|     public Battleship getDestroyedShip() { |  | ||||||
|         return destroyedShip; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the list of opponent's remaining ships after the shot. |  | ||||||
|      * |  | ||||||
|      * @return the list of opponent's remaining ships, null if the game is not yet over |  | ||||||
|      */ |  | ||||||
|     public List<Battleship> getRemainingOpponentShips() { |  | ||||||
|         return remainingOpponentShips; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the game is over. |  | ||||||
|      * |  | ||||||
|      * @return true if the game is over, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean isGameOver() { |  | ||||||
|         return remainingOpponentShips != null; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the game is won by the player. |  | ||||||
|      * |  | ||||||
|      * @return true if the game is won by the player, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean isGameWon() { |  | ||||||
|         return isGameOver() && isOwnShot(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the game is lost by the player. |  | ||||||
|      * |  | ||||||
|      * @return true if the game is lost by the player, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean isGameLost() { |  | ||||||
|         return isGameOver() && !isOwnShot(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if it's currently the player's turn. |  | ||||||
|      * |  | ||||||
|      * @return true if it's the player's turn, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean isMyTurn() { |  | ||||||
|         return isOwnShot() == shot.isHit(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns a string representation of the EffectMessage. |  | ||||||
|      * |  | ||||||
|      * @return a string representation of the EffectMessage |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public String toString() { |  | ||||||
|         return "EffectMessage{ownShot=" + ownShot + ", shot=" + shot + ", destroyedShip=" + destroyedShip + //NON-NLS |  | ||||||
|                ", remainingOpponentShips=" + remainingOpponentShips + '}'; //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the key for the informational text associated with this message. |  | ||||||
|      * |  | ||||||
|      * @return the key for the informational text |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public String getInfoTextKey() { |  | ||||||
|         if (isGameOver()) |  | ||||||
|             return isGameWon() ? "you.won.the.game" : "you.lost.the.game"; |  | ||||||
|         return isMyTurn() ? "its.your.turn" : "wait.for.opponent"; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,97 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.message.server; |  | ||||||
|  |  | ||||||
| import com.jme3.network.serializing.Serializable; |  | ||||||
| import pp.battleship.BattleshipConfig; |  | ||||||
|  |  | ||||||
| import java.util.Map; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * A message sent by the server to provide details about the game configuration. |  | ||||||
|  */ |  | ||||||
| @Serializable |  | ||||||
| public class GameDetails extends ServerMessage { |  | ||||||
|     private Map<Integer, Integer> shipNums; |  | ||||||
|     private int width; |  | ||||||
|     private int height; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Default constructor for serialization purposes. |  | ||||||
|      */ |  | ||||||
|     private GameDetails() { /* empty */ } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a GameDetails message with the specified BattleshipConfig. |  | ||||||
|      * |  | ||||||
|      * @param config the BattleshipConfig containing game configuration details |  | ||||||
|      */ |  | ||||||
|     public GameDetails(BattleshipConfig config) { |  | ||||||
|         this.shipNums = config.getShipNums(); |  | ||||||
|         this.width = config.getMapWidth(); |  | ||||||
|         this.height = config.getMapHeight(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Accepts a visitor to process this message. |  | ||||||
|      * |  | ||||||
|      * @param interpreter the visitor to process this message |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void accept(ServerInterpreter interpreter) { |  | ||||||
|         interpreter.received(this); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns a map where the keys represent ship lengths |  | ||||||
|      * and the values represent the number of ships of that length. |  | ||||||
|      * |  | ||||||
|      * @return a map of ship lengths to the number of ships |  | ||||||
|      */ |  | ||||||
|     public Map<Integer, Integer> getShipNums() { |  | ||||||
|         return shipNums; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the width of the game map. |  | ||||||
|      * |  | ||||||
|      * @return the width of the game map |  | ||||||
|      */ |  | ||||||
|     public int getWidth() { |  | ||||||
|         return width; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the height of the game map. |  | ||||||
|      * |  | ||||||
|      * @return the height of the game map |  | ||||||
|      */ |  | ||||||
|     public int getHeight() { |  | ||||||
|         return height; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns a string representation of the GameDetails message. |  | ||||||
|      * |  | ||||||
|      * @return a string representation of the GameDetails message |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public String toString() { |  | ||||||
|         return "GameDetails{" + "ships=" + getShipNums() + ", width=" + width + ", height=" + height + '}'; //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the key for the informational text associated with this message. |  | ||||||
|      * |  | ||||||
|      * @return the key for the informational text |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public String getInfoTextKey() { |  | ||||||
|         return "place.ships.in.your.map"; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,36 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.message.server; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * An interface for processing server messages. |  | ||||||
|  * Implementations of this interface can be used to handle different types of server messages. |  | ||||||
|  */ |  | ||||||
| public interface ServerInterpreter { |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles a GameDetails message received from the server. |  | ||||||
|      * |  | ||||||
|      * @param msg the GameDetails message received |  | ||||||
|      */ |  | ||||||
|     void received(GameDetails msg); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles a StartBattleMessage received from the server. |  | ||||||
|      * |  | ||||||
|      * @param msg the StartBattleMessage received |  | ||||||
|      */ |  | ||||||
|     void received(StartBattleMessage msg); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handles an EffectMessage received from the server. |  | ||||||
|      * |  | ||||||
|      * @param msg the EffectMessage received |  | ||||||
|      */ |  | ||||||
|     void received(EffectMessage msg); |  | ||||||
| } |  | ||||||
| @@ -1,39 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.message.server; |  | ||||||
|  |  | ||||||
| import com.jme3.network.AbstractMessage; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * An abstract base class for server messages used in network transfer. |  | ||||||
|  * It extends the AbstractMessage class provided by the jme3-network library. |  | ||||||
|  */ |  | ||||||
| public abstract class ServerMessage extends AbstractMessage { |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a new ServerMessage instance. |  | ||||||
|      */ |  | ||||||
|     protected ServerMessage() { |  | ||||||
|         super(true); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Accepts a visitor for processing this message. |  | ||||||
|      * |  | ||||||
|      * @param interpreter the visitor to be used for processing |  | ||||||
|      */ |  | ||||||
|     public abstract void accept(ServerInterpreter interpreter); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Gets the bundle key of the informational text to be shown at the client. |  | ||||||
|      * This key is used to retrieve the appropriate localized text for display. |  | ||||||
|      * |  | ||||||
|      * @return the bundle key of the informational text |  | ||||||
|      */ |  | ||||||
|     public abstract String getInfoTextKey(); |  | ||||||
| } |  | ||||||
| @@ -1,71 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.message.server; |  | ||||||
|  |  | ||||||
| import com.jme3.network.serializing.Serializable; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * A message sent by the server to inform clients about the start of the battle. |  | ||||||
|  */ |  | ||||||
| @Serializable |  | ||||||
| public class StartBattleMessage extends ServerMessage { |  | ||||||
|     private boolean myTurn; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Default constructor for serialization purposes. |  | ||||||
|      */ |  | ||||||
|     private StartBattleMessage() { /* empty */ } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a StartBattleMessage with the specified turn indicator. |  | ||||||
|      * |  | ||||||
|      * @param myTurn true if it's the client's turn to shoot, false otherwise |  | ||||||
|      */ |  | ||||||
|     public StartBattleMessage(boolean myTurn) { |  | ||||||
|         this.myTurn = myTurn; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Accepts a visitor to process this message. |  | ||||||
|      * |  | ||||||
|      * @param interpreter the visitor to process this message |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void accept(ServerInterpreter interpreter) { |  | ||||||
|         interpreter.received(this); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if it's the client's turn to shoot. |  | ||||||
|      * |  | ||||||
|      * @return true if it's the client's turn, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean isMyTurn() { |  | ||||||
|         return myTurn; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns a string representation of the StartBattleMessage. |  | ||||||
|      * |  | ||||||
|      * @return a string representation of the StartBattleMessage |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public String toString() { |  | ||||||
|         return "StartBattleMessage{myTurn=" + myTurn + '}'; //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the key for the informational text associated with this message. |  | ||||||
|      * |  | ||||||
|      * @return the key for the informational text |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public String getInfoTextKey() { |  | ||||||
|         return isMyTurn() ? "its.your.turn" : "wait.for.opponent"; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,324 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.model; |  | ||||||
|  |  | ||||||
| import com.jme3.network.serializing.Serializable; |  | ||||||
|  |  | ||||||
| import java.util.Collections; |  | ||||||
| import java.util.HashSet; |  | ||||||
| import java.util.Set; |  | ||||||
|  |  | ||||||
| import static java.lang.Math.max; |  | ||||||
| import static java.lang.Math.min; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Represents a battleship in the game. A battleship is characterized by its length, position, |  | ||||||
|  * rotation, and status. It can be moved, rotated, and hit during the game. This class also |  | ||||||
|  * provides methods to check for collisions with other ships and to determine whether the |  | ||||||
|  * battleship has been destroyed. |  | ||||||
|  */ |  | ||||||
| @Serializable |  | ||||||
| public class Battleship implements Item { |  | ||||||
|     /** |  | ||||||
|      * Enumeration representing the different statuses a battleship can have during the game. |  | ||||||
|      */ |  | ||||||
|     public enum Status { |  | ||||||
|         /** |  | ||||||
|          * The ship is in its normal state, not being previewed for placement. |  | ||||||
|          */ |  | ||||||
|         NORMAL, |  | ||||||
|  |  | ||||||
|         /** |  | ||||||
|          * The ship is being previewed in a valid position for placement. |  | ||||||
|          */ |  | ||||||
|         VALID_PREVIEW, |  | ||||||
|  |  | ||||||
|         /** |  | ||||||
|          * The ship is being previewed in an invalid position for placement. |  | ||||||
|          */ |  | ||||||
|         INVALID_PREVIEW |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private final int length; // The length of the battleship |  | ||||||
|     private int x;            // The x-coordinate of the battleship's position |  | ||||||
|     private int y;            // The y-coordinate of the battleship's position |  | ||||||
|     private Rotation rot;     // The rotation of the battleship |  | ||||||
|     private Status status;    // The current status of the battleship |  | ||||||
|     private final Set<IntPoint> damaged = new HashSet<>(); // The set of positions that have been hit on this ship |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Default constructor for serialization. Initializes a battleship with length 0, |  | ||||||
|      * at position (0, 0), with a default rotation of RIGHT. |  | ||||||
|      */ |  | ||||||
|     private Battleship() { |  | ||||||
|         this(0, 0, 0, Rotation.RIGHT); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a new Battleship with the specified length, position, and rotation. |  | ||||||
|      * |  | ||||||
|      * @param length the length of the battleship |  | ||||||
|      * @param x      the x-coordinate of the battleship's initial position |  | ||||||
|      * @param y      the y-coordinate of the battleship's initial position |  | ||||||
|      * @param rot    the rotation of the battleship |  | ||||||
|      */ |  | ||||||
|     public Battleship(int length, int x, int y, Rotation rot) { |  | ||||||
|         this.x = x; |  | ||||||
|         this.y = y; |  | ||||||
|         this.rot = rot; |  | ||||||
|         this.length = length; |  | ||||||
|         this.status = Status.NORMAL; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the current x-coordinate of the battleship's position. |  | ||||||
|      * |  | ||||||
|      * @return the x-coordinate of the battleship |  | ||||||
|      */ |  | ||||||
|     public int getX() { |  | ||||||
|         return x; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the current y-coordinate of the battleship's position. |  | ||||||
|      * |  | ||||||
|      * @return the y-coordinate of the battleship |  | ||||||
|      */ |  | ||||||
|     public int getY() { |  | ||||||
|         return y; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Moves the battleship to the specified coordinates. |  | ||||||
|      * |  | ||||||
|      * @param x the new x-coordinate of the battleship's position |  | ||||||
|      * @param y the new y-coordinate of the battleship's position |  | ||||||
|      */ |  | ||||||
|     public void moveTo(int x, int y) { |  | ||||||
|         this.x = x; |  | ||||||
|         this.y = y; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Moves the battleship to the specified position. |  | ||||||
|      * |  | ||||||
|      * @param pos the new position of the battleship |  | ||||||
|      */ |  | ||||||
|     public void moveTo(IntPosition pos) { |  | ||||||
|         moveTo(pos.getX(), pos.getY()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the current status of the battleship. |  | ||||||
|      * |  | ||||||
|      * @return the status of the battleship |  | ||||||
|      */ |  | ||||||
|     public Status getStatus() { |  | ||||||
|         return status; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Sets the status of the battleship. |  | ||||||
|      * |  | ||||||
|      * @param status the new status to be set for the battleship |  | ||||||
|      */ |  | ||||||
|     public void setStatus(Status status) { |  | ||||||
|         this.status = status; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the length of the battleship. |  | ||||||
|      * |  | ||||||
|      * @return the length of the battleship |  | ||||||
|      */ |  | ||||||
|     public int getLength() { |  | ||||||
|         return length; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the minimum x-coordinate that the battleship occupies based on its current position and rotation. |  | ||||||
|      * |  | ||||||
|      * @return the minimum x-coordinate of the battleship |  | ||||||
|      */ |  | ||||||
|     public int getMinX() { |  | ||||||
|         return x + min(0, (length - 1) * rot.dx()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the maximum x-coordinate that the battleship occupies based on its current position and rotation. |  | ||||||
|      * |  | ||||||
|      * @return the maximum x-coordinate of the battleship |  | ||||||
|      */ |  | ||||||
|     public int getMaxX() { |  | ||||||
|         return x + max(0, (length - 1) * rot.dx()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the minimum y-coordinate that the battleship occupies based on its current position and rotation. |  | ||||||
|      * |  | ||||||
|      * @return the minimum y-coordinate of the battleship |  | ||||||
|      */ |  | ||||||
|     public int getMinY() { |  | ||||||
|         return y + min(0, (length - 1) * rot.dy()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the maximum y-coordinate that the battleship occupies based on its current position and rotation. |  | ||||||
|      * |  | ||||||
|      * @return the maximum y-coordinate of the battleship |  | ||||||
|      */ |  | ||||||
|     public int getMaxY() { |  | ||||||
|         return y + max(0, (length - 1) * rot.dy()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the current rotation of the battleship. |  | ||||||
|      * |  | ||||||
|      * @return the rotation of the battleship |  | ||||||
|      */ |  | ||||||
|     public Rotation getRot() { |  | ||||||
|         return rot; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Sets the rotation of the battleship. |  | ||||||
|      * |  | ||||||
|      * @param rot the new rotation to be set for the battleship |  | ||||||
|      */ |  | ||||||
|     public void setRotation(Rotation rot) { |  | ||||||
|         this.rot = rot; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Rotates the battleship by 90 degrees clockwise. |  | ||||||
|      */ |  | ||||||
|     public void rotated() { |  | ||||||
|         setRotation(rot.rotate()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Attempts to hit the battleship at the specified position. |  | ||||||
|      * If the position is part of the battleship, the hit is recorded. |  | ||||||
|      * |  | ||||||
|      * @param x the x-coordinate of the position to hit |  | ||||||
|      * @param y the y-coordinate of the position to hit |  | ||||||
|      * @return true if the position is part of the battleship, false otherwise |  | ||||||
|      * @see #contains(int, int) |  | ||||||
|      */ |  | ||||||
|     public boolean hit(int x, int y) { |  | ||||||
|         if (!contains(x, y)) |  | ||||||
|             return false; |  | ||||||
|         damaged.add(new IntPoint(x, y)); |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Attempts to hit the battleship at the specified position. |  | ||||||
|      * If the position is part of the battleship, the hit is recorded. |  | ||||||
|      * This is a convenience method for {@linkplain #hit(int, int)}. |  | ||||||
|      * |  | ||||||
|      * @param position the position to hit |  | ||||||
|      * @return true if the position is part of the battleship, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean hit(IntPosition position) { |  | ||||||
|         return hit(position.getX(), position.getY()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the positions of this battleship that have been hit. |  | ||||||
|      * |  | ||||||
|      * @return a set of positions that have been hit |  | ||||||
|      * @see #hit(int, int) |  | ||||||
|      */ |  | ||||||
|     public Set<IntPoint> getDamaged() { |  | ||||||
|         return Collections.unmodifiableSet(damaged); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks whether the specified position is covered by the battleship. This method does |  | ||||||
|      * not record a hit, only checks coverage. |  | ||||||
|      * This is a convenience method for {@linkplain #contains(int, int)}. |  | ||||||
|      * |  | ||||||
|      * @param pos the position to check |  | ||||||
|      * @return true if the position is covered by the battleship, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean contains(IntPosition pos) { |  | ||||||
|         return contains(pos.getX(), pos.getY()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks whether the specified position is covered by the battleship. This method does |  | ||||||
|      * not record a hit, only checks coverage. |  | ||||||
|      * |  | ||||||
|      * @param x the x-coordinate of the position to check |  | ||||||
|      * @param y the y-coordinate of the position to check |  | ||||||
|      * @return true if the position is covered by the battleship, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean contains(int x, int y) { |  | ||||||
|         return getMinX() <= x && x <= getMaxX() && |  | ||||||
|                getMinY() <= y && y <= getMaxY(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Determines if the battleship has been completely destroyed. A battleship is considered |  | ||||||
|      * destroyed if all of its positions have been hit. |  | ||||||
|      * |  | ||||||
|      * @return true if the battleship is destroyed, false otherwise |  | ||||||
|      * @see #hit(int, int) |  | ||||||
|      */ |  | ||||||
|     public boolean isDestroyed() { |  | ||||||
|         return damaged.size() == length; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks whether this battleship collides with another battleship. Two battleships collide |  | ||||||
|      * if any of their occupied positions overlap. |  | ||||||
|      * |  | ||||||
|      * @param other the other battleship to check collision with |  | ||||||
|      * @return true if the battleships collide, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean collidesWith(Battleship other) { |  | ||||||
|         return other.getMaxX() >= getMinX() && getMaxX() >= other.getMinX() && |  | ||||||
|                other.getMaxY() >= getMinY() && getMaxY() >= other.getMinY(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns a string representation of the battleship, including its length, position, |  | ||||||
|      * and rotation. |  | ||||||
|      * |  | ||||||
|      * @return a string representation of the battleship |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public String toString() { |  | ||||||
|         return "Ship{length=" + length + ", x=" + x + ", y=" + y + ", rot=" + rot + '}';  //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Accepts a visitor that returns a value of type {@code T}. This method is part of the |  | ||||||
|      * Visitor design pattern. |  | ||||||
|      * |  | ||||||
|      * @param visitor the visitor to accept |  | ||||||
|      * @param <T>     the type of the value returned by the visitor |  | ||||||
|      * @return the value returned by the visitor |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public <T> T accept(Visitor<T> visitor) { |  | ||||||
|         return visitor.visit(this); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Accepts a visitor that does not return a value. This method is part of the |  | ||||||
|      * Visitor design pattern. |  | ||||||
|      * |  | ||||||
|      * @param visitor the visitor to accept |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void accept(VoidVisitor visitor) { |  | ||||||
|         visitor.visit(this); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,94 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.model; |  | ||||||
|  |  | ||||||
| import com.jme3.network.serializing.Serializable; |  | ||||||
|  |  | ||||||
| import java.util.Objects; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Represents a point in the two-dimensional plane with integer coordinates. |  | ||||||
|  */ |  | ||||||
| @Serializable |  | ||||||
| public final class IntPoint implements IntPosition { |  | ||||||
|     private int x; |  | ||||||
|     private int y; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Default constructor for serialization purposes. |  | ||||||
|      */ |  | ||||||
|     private IntPoint() { /* do nothing */ } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a new IntPoint with the specified coordinates. |  | ||||||
|      * |  | ||||||
|      * @param x the x-coordinate of the point |  | ||||||
|      * @param y the y-coordinate of the point |  | ||||||
|      */ |  | ||||||
|     public IntPoint(int x, int y) { |  | ||||||
|         this.x = x; |  | ||||||
|         this.y = y; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Gets the x-coordinate of the point. |  | ||||||
|      * |  | ||||||
|      * @return the x-coordinate |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public int getX() { |  | ||||||
|         return x; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Gets the y-coordinate of the point. |  | ||||||
|      * |  | ||||||
|      * @return the y-coordinate |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public int getY() { |  | ||||||
|         return y; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Indicates whether some other object is "equal to" this one. |  | ||||||
|      * |  | ||||||
|      * @param obj the reference object with which to compare |  | ||||||
|      * @return true if this object is the same as the obj argument; false otherwise |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public boolean equals(Object obj) { |  | ||||||
|         if (obj == this) return true; |  | ||||||
|         if (obj == null || obj.getClass() != this.getClass()) return false; |  | ||||||
|         var that = (IntPoint) obj; |  | ||||||
|         return this.x == that.x && |  | ||||||
|                this.y == that.y; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns a hash code value for the IntPoint. |  | ||||||
|      * |  | ||||||
|      * @return a hash code value for this object |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public int hashCode() { |  | ||||||
|         return Objects.hash(x, y); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns a string representation of the IntPoint. |  | ||||||
|      * |  | ||||||
|      * @return a string representation of the object |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public String toString() { |  | ||||||
|         return "IntPoint[" + //NON-NLS |  | ||||||
|                "x=" + x + ", " + //NON-NLS |  | ||||||
|                "y=" + y + ']'; //NON-NLS |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,28 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.model; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Interface representing a position with X and Y coordinates. |  | ||||||
|  */ |  | ||||||
| public interface IntPosition { |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the X coordinate of this position. |  | ||||||
|      * |  | ||||||
|      * @return the X coordinate. |  | ||||||
|      */ |  | ||||||
|     int getX(); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the Y coordinate of this position. |  | ||||||
|      * |  | ||||||
|      * @return the Y coordinate. |  | ||||||
|      */ |  | ||||||
|     int getY(); |  | ||||||
| } |  | ||||||
| @@ -1,31 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.model; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * An interface representing any item on a ship map. |  | ||||||
|  * It extends the IntPosition interface to provide position information. |  | ||||||
|  */ |  | ||||||
| public interface Item { |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Accepts a visitor to perform operations on the item. |  | ||||||
|      * |  | ||||||
|      * @param visitor the visitor performing operations on the item |  | ||||||
|      * @param <T>     the type of result returned by the visitor |  | ||||||
|      * @return the result of the visitor's operation on the item |  | ||||||
|      */ |  | ||||||
|     <T> T accept(Visitor<T> visitor); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Accepts a visitor to perform operations on the item without returning a result. |  | ||||||
|      * |  | ||||||
|      * @param visitor the visitor performing operations on the item |  | ||||||
|      */ |  | ||||||
|     void accept(VoidVisitor visitor); |  | ||||||
| } |  | ||||||
| @@ -1,67 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.model; |  | ||||||
|  |  | ||||||
| import java.io.Serializable; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Represents the rotation of a ship and provides functionality related to rotation. |  | ||||||
|  */ |  | ||||||
| public enum Rotation implements Serializable { |  | ||||||
|     /** |  | ||||||
|      * Represents the ship facing upwards. |  | ||||||
|      */ |  | ||||||
|     UP, |  | ||||||
|     /** |  | ||||||
|      * Represents the ship facing rightwards. |  | ||||||
|      */ |  | ||||||
|     RIGHT, |  | ||||||
|     /** |  | ||||||
|      * Represents the ship facing downwards. |  | ||||||
|      */ |  | ||||||
|     DOWN, |  | ||||||
|     /** |  | ||||||
|      * Represents the ship facing leftwards. |  | ||||||
|      */ |  | ||||||
|     LEFT; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Gets the change in x-coordinate corresponding to this rotation. |  | ||||||
|      * |  | ||||||
|      * @return the change in x-coordinate |  | ||||||
|      */ |  | ||||||
|     public int dx() { |  | ||||||
|         return switch (this) { |  | ||||||
|             case UP, DOWN -> 0; |  | ||||||
|             case RIGHT -> 1; |  | ||||||
|             case LEFT -> -1; |  | ||||||
|         }; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Gets the change in y-coordinate corresponding to this rotation. |  | ||||||
|      * |  | ||||||
|      * @return the change in y-coordinate |  | ||||||
|      */ |  | ||||||
|     public int dy() { |  | ||||||
|         return switch (this) { |  | ||||||
|             case UP -> 1; |  | ||||||
|             case LEFT, RIGHT -> 0; |  | ||||||
|             case DOWN -> -1; |  | ||||||
|         }; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Rotates the orientation clockwise and returns the next rotation. |  | ||||||
|      * |  | ||||||
|      * @return the next rotation after rotating clockwise |  | ||||||
|      */ |  | ||||||
|     public Rotation rotate() { |  | ||||||
|         return values()[(ordinal() + 1) % values().length]; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,251 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.model; |  | ||||||
|  |  | ||||||
| import pp.battleship.notification.GameEvent; |  | ||||||
| import pp.battleship.notification.GameEventBroker; |  | ||||||
| import pp.battleship.notification.ItemAddedEvent; |  | ||||||
|  |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.Collections; |  | ||||||
| import java.util.List; |  | ||||||
| import java.util.stream.Stream; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Represents a rectangular map that holds ships and registers shots fired. |  | ||||||
|  * It also supports event notification for game state changes such as item addition or removal. |  | ||||||
|  * Valid positions on this map have x-coordinates in the range of 0 to width-1 and y-coordinates |  | ||||||
|  * in the range of 0 to height-1. |  | ||||||
|  * |  | ||||||
|  * @see #getWidth() |  | ||||||
|  * @see #getHeight() |  | ||||||
|  */ |  | ||||||
| public class ShipMap { |  | ||||||
|     /** |  | ||||||
|      * A list of items (ships, shots, etc.) placed on the map. |  | ||||||
|      */ |  | ||||||
|     private final List<Item> items = new ArrayList<>(); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The broker responsible for notifying registered listeners of events |  | ||||||
|      * (such as when an item is added or removed from the map). |  | ||||||
|      * Can be null, in which case no notifications will be sent. |  | ||||||
|      */ |  | ||||||
|     private final GameEventBroker eventBroker; |  | ||||||
|  |  | ||||||
|     private final int width; |  | ||||||
|     private final int height; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs an empty map with the given dimensions. The specified event broker |  | ||||||
|      * will handle the notification of changes in the map state, such as adding or removing items. |  | ||||||
|      * Passing null as the event broker is allowed, but in that case, no notifications will occur. |  | ||||||
|      * |  | ||||||
|      * @param width       the number of columns (width) of the map |  | ||||||
|      * @param height      the number of rows (height) of the map |  | ||||||
|      * @param eventBroker the event broker used for notifying listeners, or null if event distribution is not needed |  | ||||||
|      */ |  | ||||||
|     public ShipMap(int width, int height, GameEventBroker eventBroker) { |  | ||||||
|         if (width < 1 || height < 1) |  | ||||||
|             throw new IllegalArgumentException("Invalid map size"); |  | ||||||
|         this.width = width; |  | ||||||
|         this.height = height; |  | ||||||
|         this.eventBroker = eventBroker; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Adds an item (e.g., a ship or a shot) to the map and triggers the appropriate event. |  | ||||||
|      * |  | ||||||
|      * @param item the item to be added to the map |  | ||||||
|      */ |  | ||||||
|     private void addItem(Item item) { |  | ||||||
|         items.add(item); |  | ||||||
|         notifyListeners(new ItemAddedEvent(item, this)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Adds a battleship to the map and triggers an item addition event. |  | ||||||
|      * |  | ||||||
|      * @param ship the battleship to be added to the map |  | ||||||
|      */ |  | ||||||
|     public void add(Battleship ship) { |  | ||||||
|         addItem(ship); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Registers a shot on the map, updates the state of the affected ship (if any), |  | ||||||
|      * and triggers an item addition event. |  | ||||||
|      * |  | ||||||
|      * @param shot the shot to be registered on the map |  | ||||||
|      */ |  | ||||||
|     public void add(Shot shot) { |  | ||||||
|         final Battleship ship = findShipAt(shot); |  | ||||||
|         if (ship != null) |  | ||||||
|             ship.hit(shot); |  | ||||||
|         addItem(shot); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Removes an item from the map and triggers an item removal event. |  | ||||||
|      * |  | ||||||
|      * @param item the item to be removed from the map |  | ||||||
|      */ |  | ||||||
|     public void remove(Item item) { |  | ||||||
|         items.remove(item); |  | ||||||
|         notifyListeners(new ItemAddedEvent(item, this)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Removes all items from the map and triggers corresponding removal events for each. |  | ||||||
|      */ |  | ||||||
|     public void clear() { |  | ||||||
|         new ArrayList<>(items).forEach(this::remove); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns a stream of items of a specified type (class). |  | ||||||
|      * |  | ||||||
|      * @param clazz the class type to filter items by |  | ||||||
|      * @param <T>   the type of items to return |  | ||||||
|      * @return a stream of items matching the specified class type |  | ||||||
|      */ |  | ||||||
|     private <T extends Item> Stream<T> getItems(Class<T> clazz) { |  | ||||||
|         return items.stream().filter(clazz::isInstance).map(clazz::cast); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns a stream of all battleships currently on the map. |  | ||||||
|      * |  | ||||||
|      * @return a stream of battleships |  | ||||||
|      */ |  | ||||||
|     public Stream<Battleship> getShips() { |  | ||||||
|         return getItems(Battleship.class); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns a list of all remaining battleships that have not been destroyed. |  | ||||||
|      * |  | ||||||
|      * @return a list of remaining battleships |  | ||||||
|      */ |  | ||||||
|     public List<Battleship> getRemainingShips() { |  | ||||||
|         return getShips().filter(s -> !s.isDestroyed()).toList(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns a stream of all shots fired on the map. |  | ||||||
|      * |  | ||||||
|      * @return a stream of shots |  | ||||||
|      */ |  | ||||||
|     public Stream<Shot> getShots() { |  | ||||||
|         return getItems(Shot.class); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns an unmodifiable list of all items currently on the map. |  | ||||||
|      * |  | ||||||
|      * @return an unmodifiable list of all items |  | ||||||
|      */ |  | ||||||
|     public List<Item> getItems() { |  | ||||||
|         return Collections.unmodifiableList(items); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the width (number of columns) of the map. |  | ||||||
|      * |  | ||||||
|      * @return the width of the map |  | ||||||
|      */ |  | ||||||
|     public int getWidth() { |  | ||||||
|         return width; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the height (number of rows) of the map. |  | ||||||
|      * |  | ||||||
|      * @return the height of the map |  | ||||||
|      */ |  | ||||||
|     public int getHeight() { |  | ||||||
|         return height; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the given ship is in a valid position (within the map bounds and non-colliding with other ships). |  | ||||||
|      * |  | ||||||
|      * @param ship the battleship to validate |  | ||||||
|      * @return true if the ship's position is valid, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean isValid(Battleship ship) { |  | ||||||
|         return isValid(ship.getMinX(), ship.getMinY()) && |  | ||||||
|                isValid(ship.getMaxX(), ship.getMaxY()) && |  | ||||||
|                getShips().filter(s -> s != ship).noneMatch(ship::collidesWith); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Finds a battleship at the specified coordinates. |  | ||||||
|      * |  | ||||||
|      * @param x the x-coordinate of the position |  | ||||||
|      * @param y the y-coordinate of the position |  | ||||||
|      * @return the ship at the specified coordinates, or null if none is found |  | ||||||
|      */ |  | ||||||
|     public Battleship findShipAt(int x, int y) { |  | ||||||
|         return getShips().filter(ship -> ship.contains(x, y)) |  | ||||||
|                          .findAny() |  | ||||||
|                          .orElse(null); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Finds a battleship at the specified position. This is a convenience method. |  | ||||||
|      * |  | ||||||
|      * @param position the position within the map |  | ||||||
|      * @return the ship at the specified position, or null if none is found |  | ||||||
|      */ |  | ||||||
|     public Battleship findShipAt(IntPosition position) { |  | ||||||
|         return findShipAt(position.getX(), position.getY()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Validates whether the specified position is within the map boundaries. |  | ||||||
|      * |  | ||||||
|      * @param pos the position to validate |  | ||||||
|      * @return true if the position is within the map, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean isValid(IntPosition pos) { |  | ||||||
|         return isValid(pos.getX(), pos.getY()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the specified coordinates are within the map boundaries. |  | ||||||
|      * |  | ||||||
|      * @param x the x-coordinate to validate |  | ||||||
|      * @param y the y-coordinate to validate |  | ||||||
|      * @return true if the coordinates are valid, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean isValid(int x, int y) { |  | ||||||
|         return x >= 0 && x < width && |  | ||||||
|                y >= 0 && y < height; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns a string representation of the ship map. |  | ||||||
|      * |  | ||||||
|      * @return a string representation of the ship map |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public String toString() { |  | ||||||
|         return "ShipMap{" + items + '}'; //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Notifies all registered listeners about a game event if the event broker is available. |  | ||||||
|      * |  | ||||||
|      * @param event the event to be distributed to listeners |  | ||||||
|      */ |  | ||||||
|     private void notifyListeners(GameEvent event) { |  | ||||||
|         if (eventBroker != null) |  | ||||||
|             eventBroker.notifyListeners(event); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,140 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.model; |  | ||||||
|  |  | ||||||
| import com.jme3.network.serializing.Serializable; |  | ||||||
|  |  | ||||||
| import java.util.Objects; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Represents a shot in the Battleship game. |  | ||||||
|  * A shot is defined by its coordinates and whether it was a hit or miss. |  | ||||||
|  */ |  | ||||||
| @Serializable |  | ||||||
| public final class Shot implements Item, IntPosition { |  | ||||||
|     private int x; |  | ||||||
|     private int y; |  | ||||||
|     private boolean hit; |  | ||||||
|  |  | ||||||
|     // Private no-arg constructor for serialization purposes |  | ||||||
|     private Shot() {} |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates a new shot. |  | ||||||
|      * |  | ||||||
|      * @param x   the x-coordinate of the shot |  | ||||||
|      * @param y   the y-coordinate of the shot |  | ||||||
|      * @param hit indicates whether the shot was a hit |  | ||||||
|      */ |  | ||||||
|     public Shot(int x, int y, boolean hit) { |  | ||||||
|         this.x = x; |  | ||||||
|         this.y = y; |  | ||||||
|         this.hit = hit; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Creates a new shot. |  | ||||||
|      * |  | ||||||
|      * @param pos the position of the shot |  | ||||||
|      * @param hit indicates whether the shot was a hit |  | ||||||
|      */ |  | ||||||
|     public Shot(IntPosition pos, boolean hit) { |  | ||||||
|         this(pos.getX(), pos.getY(), hit); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Gets the x-coordinate of the shot. |  | ||||||
|      * |  | ||||||
|      * @return the x-coordinate of the shot |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public int getX() { |  | ||||||
|         return x; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Gets the y-coordinate of the shot. |  | ||||||
|      * |  | ||||||
|      * @return the y-coordinate of the shot |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public int getY() { |  | ||||||
|         return y; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the shot was a hit. |  | ||||||
|      * |  | ||||||
|      * @return true if the shot was a hit, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean isHit() { |  | ||||||
|         return hit; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if this shot is equal to another object. |  | ||||||
|      * Two shots are considered equal if they have the same coordinates and hit status. |  | ||||||
|      * |  | ||||||
|      * @param obj the object to compare with |  | ||||||
|      * @return true if the objects are equal, false otherwise |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public boolean equals(Object obj) { |  | ||||||
|         if (obj == this) return true; |  | ||||||
|         if (obj == null || obj.getClass() != this.getClass()) return false; |  | ||||||
|         var that = (Shot) obj; |  | ||||||
|         return this.x == that.x && |  | ||||||
|                this.y == that.y && |  | ||||||
|                this.hit == that.hit; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Computes the hash code of this shot. |  | ||||||
|      * |  | ||||||
|      * @return the hash code of this shot |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public int hashCode() { |  | ||||||
|         return Objects.hash(x, y, hit); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns a string representation of the shot. |  | ||||||
|      * |  | ||||||
|      * @return a string representation of the shot |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public String toString() { |  | ||||||
|         return "Shot[" + //NON-NLS |  | ||||||
|                "x=" + x + ", " + //NON-NLS |  | ||||||
|                "y=" + y + ", " + //NON-NLS |  | ||||||
|                "hit=" + hit + ']'; //NON-NLS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Accepts a visitor with a return value. |  | ||||||
|      * |  | ||||||
|      * @param visitor the visitor to accept |  | ||||||
|      * @param <T>     the type of the return value |  | ||||||
|      * @return the result of the visitor's visit method |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public <T> T accept(Visitor<T> visitor) { |  | ||||||
|         return visitor.visit(this); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Accepts a visitor without a return value. |  | ||||||
|      * |  | ||||||
|      * @param visitor the visitor to accept |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void accept(VoidVisitor visitor) { |  | ||||||
|         visitor.visit(this); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,31 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.model; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * An interface for implementing the Visitor pattern for different types of elements in the Battleship model. |  | ||||||
|  * |  | ||||||
|  * @param <T> the type of result returned by the visit methods |  | ||||||
|  */ |  | ||||||
| public interface Visitor<T> { |  | ||||||
|     /** |  | ||||||
|      * Visits a Shot element. |  | ||||||
|      * |  | ||||||
|      * @param shot the Shot element to visit |  | ||||||
|      * @return the result of visiting the Shot element |  | ||||||
|      */ |  | ||||||
|     T visit(Shot shot); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Visits a Battleship element. |  | ||||||
|      * |  | ||||||
|      * @param ship the Battleship element to visit |  | ||||||
|      * @return the result of visiting the Battleship element |  | ||||||
|      */ |  | ||||||
|     T visit(Battleship ship); |  | ||||||
| } |  | ||||||
| @@ -1,28 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.model; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * An interface for implementing the Visitor pattern for different types of elements in the Battleship model |  | ||||||
|  * without returning any result. |  | ||||||
|  */ |  | ||||||
| public interface VoidVisitor { |  | ||||||
|     /** |  | ||||||
|      * Visits a Shot element. |  | ||||||
|      * |  | ||||||
|      * @param shot the Shot element to visit |  | ||||||
|      */ |  | ||||||
|     void visit(Shot shot); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Visits a Battleship element. |  | ||||||
|      * |  | ||||||
|      * @param ship the Battleship element to visit |  | ||||||
|      */ |  | ||||||
|     void visit(Battleship ship); |  | ||||||
| } |  | ||||||
| @@ -1,51 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.model.dto; |  | ||||||
|  |  | ||||||
| import pp.battleship.model.Battleship; |  | ||||||
| import pp.battleship.model.Rotation; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * A class representing data transfer objects of battleships for JSON serialization and deserialization. |  | ||||||
|  */ |  | ||||||
| class BattleshipDTO { |  | ||||||
|     private final int length;   // The length of the battleship |  | ||||||
|     private final int x;        // The x-coordinate of the battleship's position |  | ||||||
|     private final int y;        // The y-coordinate of the battleship's position |  | ||||||
|     private final Rotation rot; // The rotation of the battleship |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a BattleshipDTO object from a Battleship object. |  | ||||||
|      * |  | ||||||
|      * @param ship the Battleship object to be converted |  | ||||||
|      */ |  | ||||||
|     BattleshipDTO(Battleship ship) { |  | ||||||
|         length = ship.getLength(); |  | ||||||
|         x = ship.getX(); |  | ||||||
|         y = ship.getY(); |  | ||||||
|         rot = ship.getRot(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Gets the length of the battleship. |  | ||||||
|      * |  | ||||||
|      * @return the length |  | ||||||
|      */ |  | ||||||
|     public int getLength() { |  | ||||||
|         return length; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Converts this BattleshipDTO object to a Battleship object. |  | ||||||
|      * |  | ||||||
|      * @return the Battleship object |  | ||||||
|      */ |  | ||||||
|     Battleship toBattleship() { |  | ||||||
|         return new Battleship(length, x, y, rot); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,117 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.model.dto; |  | ||||||
|  |  | ||||||
| import com.google.gson.Gson; |  | ||||||
| import com.google.gson.GsonBuilder; |  | ||||||
| import com.google.gson.JsonParseException; |  | ||||||
| import pp.battleship.message.server.GameDetails; |  | ||||||
| import pp.battleship.model.Battleship; |  | ||||||
| import pp.battleship.model.ShipMap; |  | ||||||
|  |  | ||||||
| import java.io.File; |  | ||||||
| import java.io.FileReader; |  | ||||||
| import java.io.FileWriter; |  | ||||||
| import java.io.IOException; |  | ||||||
| import java.lang.System.Logger; |  | ||||||
| import java.lang.System.Logger.Level; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * A class representing data transfer objects of ship maps for JSON serialization and deserialization. |  | ||||||
|  */ |  | ||||||
| public class ShipMapDTO { |  | ||||||
|     // Logger instance for logging messages |  | ||||||
|     static final Logger LOGGER = System.getLogger(ShipMapDTO.class.getName()); |  | ||||||
|  |  | ||||||
|     // Width of the ship map |  | ||||||
|     private int width; |  | ||||||
|  |  | ||||||
|     // Height of the ship map |  | ||||||
|     private int height; |  | ||||||
|  |  | ||||||
|     // List of ships in the map |  | ||||||
|     private List<BattleshipDTO> ships; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Constructs a ShipMapDTO object from a ShipMap object. |  | ||||||
|      * |  | ||||||
|      * @param map the ShipMap object to be converted |  | ||||||
|      */ |  | ||||||
|     public ShipMapDTO(ShipMap map) { |  | ||||||
|         this.width = map.getWidth(); |  | ||||||
|         this.height = map.getHeight(); |  | ||||||
|         this.ships = map.getShips().map(BattleshipDTO::new).toList(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Checks if the current ship map fits the game details provided. |  | ||||||
|      * |  | ||||||
|      * @param details the game details to be matched against |  | ||||||
|      * @return true if the ship map fits the game details, false otherwise |  | ||||||
|      */ |  | ||||||
|     public boolean fits(GameDetails details) { |  | ||||||
|         if (width != details.getWidth() || height != details.getHeight()) |  | ||||||
|             return false; |  | ||||||
|         int numShips = details.getShipNums().values().stream().mapToInt(Integer::intValue).sum(); |  | ||||||
|         if (numShips != ships.size()) |  | ||||||
|             return false; |  | ||||||
|         for (var e : details.getShipNums().entrySet()) { |  | ||||||
|             final int shipLength = e.getKey(); |  | ||||||
|             final int requiredNum = e.getValue(); |  | ||||||
|             final int actualNum = (int) ships.stream() |  | ||||||
|                                              .filter(s -> s.getLength() == shipLength) |  | ||||||
|                                              .count(); |  | ||||||
|             if (requiredNum != actualNum) |  | ||||||
|                 return false; |  | ||||||
|         } |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the ships stored in this DTO. |  | ||||||
|      * |  | ||||||
|      * @return the ships stored in this DTO |  | ||||||
|      */ |  | ||||||
|     public List<Battleship> getShips() { |  | ||||||
|         return ships.stream().map(BattleshipDTO::toBattleship).toList(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Saves the current ShipMapDTO to a file in JSON format. |  | ||||||
|      * |  | ||||||
|      * @param file the file to which the ShipMapDTO will be saved |  | ||||||
|      * @throws IOException if an I/O error occurs |  | ||||||
|      */ |  | ||||||
|     public void saveTo(File file) throws IOException { |  | ||||||
|         try (FileWriter writer = new FileWriter(file)) { |  | ||||||
|             final Gson gson = new GsonBuilder().setPrettyPrinting().create(); |  | ||||||
|             final String json = gson.toJson(this); |  | ||||||
|             LOGGER.log(Level.DEBUG, "JSON of map: {0}", json); //NON-NLS |  | ||||||
|             writer.write(json); |  | ||||||
|             LOGGER.log(Level.INFO, "JSON written to {0}", file.getAbsolutePath()); //NON-NLS |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Loads a ShipMapDTO from a file containing JSON data. |  | ||||||
|      * |  | ||||||
|      * @param file the file from which the ShipMapDTO will be loaded |  | ||||||
|      * @return the loaded ShipMapDTO object |  | ||||||
|      * @throws IOException if an I/O error occurs or if the JSON is invalid |  | ||||||
|      */ |  | ||||||
|     public static ShipMapDTO loadFrom(File file) throws IOException { |  | ||||||
|         try (FileReader reader = new FileReader(file)) { |  | ||||||
|             final Gson gson = new Gson(); |  | ||||||
|             return gson.fromJson(reader, ShipMapDTO.class); |  | ||||||
|         } |  | ||||||
|         catch (JsonParseException e) { |  | ||||||
|             throw new IOException(e.getLocalizedMessage()); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,23 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.notification; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Event when an item is added to a map. |  | ||||||
|  */ |  | ||||||
| public record ClientStateEvent() implements GameEvent { |  | ||||||
|     /** |  | ||||||
|      * Notifies the game event listener of this event. |  | ||||||
|      * |  | ||||||
|      * @param listener the game event listener |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void notifyListener(GameEventListener listener) { |  | ||||||
|         listener.receivedEvent(this); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,20 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.notification; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * An interface used for all game events. |  | ||||||
|  */ |  | ||||||
| public interface GameEvent { |  | ||||||
|     /** |  | ||||||
|      * Notifies the game event listener of the event. |  | ||||||
|      * |  | ||||||
|      * @param listener the game event listener to be notified |  | ||||||
|      */ |  | ||||||
|     void notifyListener(GameEventListener listener); |  | ||||||
| } |  | ||||||
| @@ -1,20 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.notification; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Defines a broker for distributing game events to registered listeners. |  | ||||||
|  */ |  | ||||||
| public interface GameEventBroker { |  | ||||||
|     /** |  | ||||||
|      * Notifies all registered listeners about the specified game event. |  | ||||||
|      * |  | ||||||
|      * @param event the game event to be broadcast to listeners |  | ||||||
|      */ |  | ||||||
|     void notifyListeners(GameEvent event); |  | ||||||
| } |  | ||||||
| @@ -1,48 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.notification; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Listener interface for all events implemented by subclasses of {@linkplain pp.battleship.notification.GameEvent}. |  | ||||||
|  */ |  | ||||||
| public interface GameEventListener { |  | ||||||
|     /** |  | ||||||
|      * Indicates that an item has been destroyed |  | ||||||
|      * |  | ||||||
|      * @param event the received event |  | ||||||
|      */ |  | ||||||
|     default void receivedEvent(ItemRemovedEvent event) { /* do nothing */ } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Indicates that an item has been added to a map. |  | ||||||
|      * |  | ||||||
|      * @param event the received event |  | ||||||
|      */ |  | ||||||
|     default void receivedEvent(ItemAddedEvent event) { /* do nothing */ } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Indicates that an info text shall be shown. |  | ||||||
|      * |  | ||||||
|      * @param event the received event |  | ||||||
|      */ |  | ||||||
|     default void receivedEvent(InfoTextEvent event) { /* do nothing */ } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Indicates that a sound shall be played. |  | ||||||
|      * |  | ||||||
|      * @param event the received event |  | ||||||
|      */ |  | ||||||
|     default void receivedEvent(SoundEvent event) { /* do nothing */ } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Indicates that the client's state has changed. |  | ||||||
|      * |  | ||||||
|      * @param event the received event |  | ||||||
|      */ |  | ||||||
|     default void receivedEvent(ClientStateEvent event) { /* do nothing */ } |  | ||||||
| } |  | ||||||
| @@ -1,25 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.notification; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Event when an item is added to a map. |  | ||||||
|  * |  | ||||||
|  * @param key the bundle key for the message |  | ||||||
|  */ |  | ||||||
| public record InfoTextEvent(String key) implements GameEvent { |  | ||||||
|     /** |  | ||||||
|      * Notifies the game event listener of this event. |  | ||||||
|      * |  | ||||||
|      * @param listener the game event listener |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void notifyListener(GameEventListener listener) { |  | ||||||
|         listener.receivedEvent(this); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,29 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.notification; |  | ||||||
|  |  | ||||||
| import pp.battleship.model.Item; |  | ||||||
| import pp.battleship.model.ShipMap; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Event when an item is added to a map. |  | ||||||
|  * |  | ||||||
|  * @param item the added item |  | ||||||
|  * @param map  the map that got the additional item |  | ||||||
|  */ |  | ||||||
| public record ItemAddedEvent(Item item, ShipMap map) implements GameEvent { |  | ||||||
|     /** |  | ||||||
|      * Notifies the game event listener of this event. |  | ||||||
|      * |  | ||||||
|      * @param listener the game event listener |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void notifyListener(GameEventListener listener) { |  | ||||||
|         listener.receivedEvent(this); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,28 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.notification; |  | ||||||
|  |  | ||||||
| import pp.battleship.model.Item; |  | ||||||
| import pp.battleship.model.ShipMap; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Event when an item gets removed. |  | ||||||
|  * |  | ||||||
|  * @param item the destroyed item |  | ||||||
|  */ |  | ||||||
| public record ItemRemovedEvent(Item item, ShipMap map) implements GameEvent { |  | ||||||
|     /** |  | ||||||
|      * Notifies the game event listener of this event. |  | ||||||
|      * |  | ||||||
|      * @param listener the game event listener |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void notifyListener(GameEventListener listener) { |  | ||||||
|         listener.receivedEvent(this); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,26 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.notification; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Enumeration representing different types of sounds used in the game. |  | ||||||
|  */ |  | ||||||
| public enum Sound { |  | ||||||
|     /** |  | ||||||
|      * Sound of an explosion. |  | ||||||
|      */ |  | ||||||
|     EXPLOSION, |  | ||||||
|     /** |  | ||||||
|      * Sound of a splash. |  | ||||||
|      */ |  | ||||||
|     SPLASH, |  | ||||||
|     /** |  | ||||||
|      * Sound of a ship being destroyed. |  | ||||||
|      */ |  | ||||||
|     DESTROYED_SHIP |  | ||||||
| } |  | ||||||
| @@ -1,26 +0,0 @@ | |||||||
| //////////////////////////////////////// |  | ||||||
| // Programming project code |  | ||||||
| // UniBw M, 2022, 2023, 2024 |  | ||||||
| // www.unibw.de/inf2 |  | ||||||
| // (c) Mark Minas (mark.minas@unibw.de) |  | ||||||
| //////////////////////////////////////// |  | ||||||
|  |  | ||||||
| package pp.battleship.notification; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Event when an item is added to a map. |  | ||||||
|  * |  | ||||||
|  * @param sound the sound to be played |  | ||||||
|  */ |  | ||||||
| public record SoundEvent(Sound sound) implements GameEvent { |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Notifies the game event listener of this event. |  | ||||||
|      * |  | ||||||
|      * @param listener the game event listener |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void notifyListener(GameEventListener listener) { |  | ||||||
|         listener.receivedEvent(this); |  | ||||||
|     } |  | ||||||
| } |  | ||||||