commit ca81dfc2ea62448ca232abba8a0513df45fb0f31 parent c26029dbbe65f55e0f83571ba47f6090c556cadf Author: A Koens <[email protected]> Date: Wed, 18 Jan 2023 15:04:19 +0100 Squashed commit of the following: commit 2373586211880a57cf72da87fe923a3f326a2f78 Merge: 15838d6 8c2855a Author: A Koens <[email protected]> Date: Wed Jan 18 15:03:54 2023 +0100 Update heuristics.json commit 15838d6b1a8625e9292463bc2aa68ff6f0a085c3 Author: A Koens <[email protected]> Date: Wed Jan 18 15:01:07 2023 +0100 Update heuristics.json commit 8c2855a53ea7e39fb42165ead4493588ce16f891 Author: yljh1 <[email protected]> Date: Wed Jan 18 15:00:42 2023 +0100 Update heuristics.json commit de5602b923537c663302a7b8820a4eaa9552cbd4 Author: A Koens <[email protected]> Date: Wed Jan 18 14:38:11 2023 +0100 Data for games Now prints data for games. commit 311732363fcab9d5cfcb8836e160b7b1e70c7452 Author: A Koens <[email protected]> Date: Tue Jan 17 12:57:33 2023 +0100 Update Headless.java commit e74e7deacff7e305cee980d08d2317b41adb604d Author: A Koens <[email protected]> Date: Tue Jan 17 12:46:59 2023 +0100 nieuw print winner commit e2136e109a72242117b2e84459726d4ed2266e68 Merge: bf9c4e0 9225f45 Author: A Koens <[email protected]> Date: Tue Jan 17 12:35:57 2023 +0100 Merge branch 'othello-basis' of https://github.com/Akoens/ISYGameClient into othello-basis commit bf9c4e025715e153ea81ec05f7e0f3086bb5dacf Author: A Koens <[email protected]> Date: Tue Jan 17 12:35:14 2023 +0100 Ai Loop commit 9225f459ec5ff442a7ae950c9f718665bfd8cc6c Author: Johs Mulder <[email protected]> Date: Mon Jan 16 22:30:49 2023 +0100 "7 Heuristieken, waaronde rmijn persoonlijke" commit 6f09ad7593f09af9d3476ba9537efd5beca786a5 Author: A Koens <[email protected]> Date: Mon Jan 16 22:00:23 2023 +0100 Heuristics Ai player Ai is now a player. Headless simulation now complete. commit 4e34e63c5c6cc4d216003bf1af9bf6b536b33e5d Author: A Koens <[email protected]> Date: Thu Jan 12 13:44:57 2023 +0100 Game Refactor Removed sperate board classes for different games. Incorporated the necessary functions into the game interface. commit 285b09baead5823a9d81e8d0a5a3ff02c551c7aa Author: A Koens <[email protected]> Date: Wed Jan 4 16:32:57 2023 +0100 Correction to stage changing GameSelector and GameSelectorMenu name discrepancies have been corrected. commit fa3a9d63d4342d4c415f79f5b609bec8456e1f35 Author: TKoekoek <[email protected]> Date: Wed Jan 4 14:59:40 2023 +0100 Othello Singleplayer commit 8178dc482d0ef9fe0829457157882e54bbf6804b Author: TKoekoek <[email protected]> Date: Fri Dec 23 11:22:15 2022 +0100 Othello Update Diffstat:
54 files changed, 2570 insertions(+), 1568 deletions(-)
diff --git a/heuristics.json b/heuristics.json @@ -0,0 +1,133 @@ +{ + "H1Linear": [ + [2, 1, 1, 1, 1, 1, 1, 2], + [1, 1, 1, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [2, 1, 1, 1, 1, 1, 1, 2] + ], + + "H2Linear": [ + [2, -1, 1, 1, 1, 1, -1, 2], + [-1, -1, 1, 1, 1, 1, -1, -1], + [1, 1, 1, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [-1, -1, 1, 1, 1, 1, -1, -1], + [2, -1, 1, 1, 1, 1, -1, 2] + ], + + "H3Linear": [ + [3, -2, 2, 2, 2, 2, -2, 3], + [-2, -2, 1, 1, 1, 1, -2, -2], + [2, 1, 1, 1, 1, 1, 1, 2], + [2, 1, 1, 1, 1, 1, 1, 2], + [2, 1, 1, 1, 1, 1, 1, 2], + [2, 1, 1, 1, 1, 1, 1, 2], + [-2, -2, 1, 1, 1, 1, -2, -2], + [3, -2, 2, 2, 2, 2, -2, 3] + ], + + "H4Linear": [ + [4, -3, 2, 2, 2, 2, -3, 4], + [-3, -3, 1, 1, 1, 1, -3, -3], + [2, 1, 1, 1, 1, 1, 1, 2], + [2, 1, 1, 1, 1, 1, 1, 2], + [2, 1, 1, 1, 1, 1, 1, 2], + [2, 1, 1, 1, 1, 1, 1, 2], + [-3, -3, 1, 1, 1, 1, -3, -3], + [4, -3, 2, 2, 2, 2, -3, 4] + ], + + "H1Exponential": [ + [5, 1, 1, 1, 1, 1, 1, 5], + [1, 1, 1, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [5, 1, 1, 1, 1, 1, 1, 5] + ], + + "H2Exponential": [ + [5, -5, 1, 1, 1, 1, -5, 5], + [-5, -5, 1, 1, 1, 1, -5, -5], + [1, 1, 1, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [-5, -5, 1, 1, 1, 1, -5, -5], + [5, -5, 1, 1, 1, 1, -5, 5] + ], + + "H3Exponential": [ + [25, -25, 5, 5, 5, 5, -25, 25], + [-25, -25, 1, 1, 1, 1, -25, -25], + [5, 1, 1, 1, 1, 1, 1, 5], + [5, 1, 1, 1, 1, 1, 1, 5], + [5, 1, 1, 1, 1, 1, 1, 5], + [5, 1, 1, 1, 1, 1, 1, 5], + [-25, -25, 1, 1, 1, 1, -25, -25], + [25, -25, 5, 5, 5, 5, -25, 25] + ], + + "Johs": [ + [10, -99, 5, 5, 5, 5, -99, 10], + [-99, -99, 3, 1, 1, 3, -99, -99], + [5, 3, 3, 1, 1, 3, 3, 5], + [5, 1, 1, 1, 1, 1, 1, 5], + [5, 1, 1, 1, 1, 1, 1, 5], + [5, 3, 3, 1, 1, 3, 3, 5], + [-99, -99, 3, 1, 1, 3, -99, -99], + [10, -99, 5, 5, 5, 5, -99, 10] + ], + + "Arjan": [ + [10, -3, 5, -2, -2, -2, 3, 10], + [-3, -3, 5, 1, 1, 1, 3, 3], + [5, 5, 5, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [1, 1, 1, 1, 1, 1, 1, 1], + [3, 3, 1, 1, 1, 1, 1, 1], + [-4, 3, -2, -2, -2, 2, 3, -4] + ], + + "Yannick": [ + [10, 2, 5, 5, 5, 5, 2, 10], + [2, 2, 1, 1, 1, 1, 2, 2], + [5, 1, 1, 1, 1, 1, 1, 5], + [5, 1, 1, 1, 1, 1, 1, 5], + [5, 1, 1, 1, 1, 1, 1, 5], + [5, 1, 1, 1, 1, 1, 1, 5], + [2, 2, 1, 1, 1, 1, 2, 2], + [10, 2, 5, 5, 5, 5, 2, 10] + ], + + "Friedel": [ + [4, 3, 3, 3, 3, 3, 3, 4], + [3, 2, 2, 2, 2, 2, 2, 3], + [3, 2, 1, 1, 1, 1, 2, 3], + [3, 2, 1, 1, 1, 1, 2, 3], + [3, 2, 1, 1, 1, 1, 2, 3], + [3, 2, 1, 1, 1, 1, 2, 3], + [3, 2, 2, 2, 2, 2, 2, 3], + [4, 3, 3, 3, 3, 3, 3, 4] + ], + + "nog_een": [ + [3, 2, 2, 2, 2, 2, 2, 3], + [2, 1, 1, 1, 1, 1, 1, 2], + [2, 1, 1, 1, 1, 1, 1, 2], + [2, 1, 1, 1, 1, 1, 1, 2], + [2, 1, 1, 1, 1, 1, 1, 2], + [2, 1, 1, 1, 1, 1, 1, 2], + [2, 1, 1, 1, 1, 1, 1, 2], + [3, 2, 2, 2, 2, 2, 2, 3] + ] +} diff --git a/pom.xml b/pom.xml @@ -20,11 +20,13 @@ <artifactId>javafx-controls</artifactId> <version>19</version> </dependency> + <dependency> <groupId>org.openjfx</groupId> <artifactId>javafx-fxml</artifactId> <version>19</version> </dependency> + <dependency> <groupId>org.controlsfx</groupId> <artifactId>controlsfx</artifactId> @@ -35,7 +37,7 @@ <dependency> <groupId>com.jfoenix</groupId> <artifactId>jfoenix</artifactId> - <version>9.0.10</version> + <version>9.0.1</version> </dependency> <!-- https://mvnrepository.com/artifact/de.jensd/fontawesomefx-commons --> @@ -52,14 +54,13 @@ <version>2.2.0-9.1.2</version> </dependency> - <!-- https://mvnrepository.com/artifact/de.jensd/fontawesomefx-fontawesome --> + <!-- GSON Dependencies --> <dependency> - <groupId>de.jensd</groupId> - <artifactId>fontawesomefx-fontawesome</artifactId> - <version>4.7.0-9.1.2</version> + <groupId>com.google.guava</groupId> + <artifactId>guava</artifactId> + <version>31.1-jre</version> </dependency> - <!-- GSON Dependencies --> <dependency> <groupId>com.google.code.gson</groupId> <artifactId>gson</artifactId> @@ -74,12 +75,25 @@ <version>${junit.version}</version> <scope>test</scope> </dependency> + <dependency> <groupId>org.junit.jupiter</groupId> <artifactId>junit-jupiter-engine</artifactId> <version>${junit.version}</version> <scope>test</scope> </dependency> + + <!-- Data Export --> + <dependency> + <groupId>org.apache.poi</groupId> + <artifactId>poi</artifactId> + <version>5.2.0</version> + </dependency> + <dependency> + <groupId>org.apache.poi</groupId> + <artifactId>poi-ooxml</artifactId> + <version>5.2.0</version> + </dependency> </dependencies> <build> @@ -106,6 +120,7 @@ <name>Hanze Hogeschool Groningen</name> <url>https://www.hanze.nl</url> </organization> + <developers> <developer> <id>afko</id> @@ -119,6 +134,29 @@ </roles> <timezone>Nederland/Amsterdam</timezone> </developer> + <developer> + <id>jomu</id> + <name>Johs Mulder</name> + <email>[email protected]</email> + <organization>Hanze Hogeschool Groningen</organization> + <organizationUrl>https://www.hanze.nl</organizationUrl> + <roles> + <role>Designer</role> + <role>Software Engineer</role> + </roles> + <timezone>Nederland/Amsterdam</timezone> + </developer> + <developer> + <id>frsh</id> + <name>Friedel Schön</name> + <email>[email protected]</email> + <organization>Hanze Hogeschool Groningen</organization> + <organizationUrl>https://www.hanze.nl</organizationUrl> + <roles> + <role>Software Engineer</role> + </roles> + <timezone>Nederland/Amsterdam</timezone> + </developer> </developers> <contributors> <contributor> @@ -127,7 +165,7 @@ <organization>Hanze Hogeschool Groningen</organization> <organizationUrl>http://hanze.nl</organizationUrl> <roles> - <role>product-owner</role> + <role>stake-holder</role> <role>tester</role> </roles> <timezone>Nederland/Amsterdam</timezone> diff --git a/src/main/java/module-info.java b/src/main/java/module-info.java @@ -6,7 +6,6 @@ module nl.isygameclient { requires org.controlsfx.controls; // Material Design - requires de.jensd.fx.glyphs.fontawesome; requires de.jensd.fx.glyphs.materialicons; requires de.jensd.fx.glyphs.commons; requires com.jfoenix; @@ -16,23 +15,30 @@ module nl.isygameclient { exports nl.isygameclient; // Controllers - exports nl.isygameclient.controllers.Games.TicTacToe; - exports nl.isygameclient.controllers.GameSeletorMenu; + exports nl.isygameclient.controllers.games.tictactoe; + exports nl.isygameclient.controllers.game_selector; // Models exports nl.isygameclient.models; exports nl.isygameclient.util; - exports nl.isygameclient.models.Settings; + exports nl.isygameclient.models.settings; + + exports nl.isygameclient.models.games.tictactoe; + exports nl.isygameclient.models.games.othello; + exports nl.isygameclient.models.board; - opens nl.isygameclient.util to com.google.gson; opens nl.isygameclient to javafx.fxml; - opens nl.isygameclient.controllers.Games to javafx.fxml; - opens nl.isygameclient.controllers.GameSeletorMenu to javafx.fxml; - opens nl.isygameclient.controllers.Games.TicTacToe to javafx.fxml; + opens nl.isygameclient.controllers.games to javafx.fxml; + opens nl.isygameclient.controllers.game_selector to javafx.fxml; + opens nl.isygameclient.controllers.games.tictactoe to javafx.fxml; + opens nl.isygameclient.controllers.games.othello to javafx.fxml; + opens nl.isygameclient.util to com.google.gson; opens nl.isygameclient.models to com.google.gson; - opens nl.isygameclient.models.Settings to com.google.gson; - exports nl.isygameclient.models.Games; - opens nl.isygameclient.models.Games to com.google.gson; -} -\ No newline at end of file + opens nl.isygameclient.models.board to com.google.gson; + opens nl.isygameclient.models.settings to com.google.gson; + opens nl.isygameclient.models.games.othello to com.google.gson; + opens nl.isygameclient.models.games.tictactoe to com.google.gson; + +} diff --git a/src/main/java/nl/isygameclient/Application.java b/src/main/java/nl/isygameclient/Application.java @@ -29,12 +29,14 @@ public class Application extends javafx.application.Application { StageHandler stageHandler = StageHandler.get(); stageHandler.loadStages(); - String stageName = "GameSelectorMenu"; - stageHandler.changeSceneOfStage(stageName, "views/GameSelectorMenu/GameSelectorMenu.fxml"); + + String stageName = "GameSelector"; + stageHandler.changeSceneOfStage(stageName, "views/game_selector/GameSelector.fxml"); + stageHandler.focusStage(stageName); stageHandler.getStage(stageName).setOnCloseRequest((e) -> { stageHandler.saveStages(); }); } -} -\ No newline at end of file +} diff --git a/src/main/java/nl/isygameclient/Headless.java b/src/main/java/nl/isygameclient/Headless.java @@ -0,0 +1,109 @@ +package nl.isygameclient; + +import com.google.gson.Gson; +import com.google.gson.JsonElement; +import com.google.gson.reflect.TypeToken; +import nl.isygameclient.models.Ai; +import nl.isygameclient.models.Game; +import nl.isygameclient.models.Player; +import nl.isygameclient.models.PlayerManager; +import nl.isygameclient.models.games.othello.Othello; +import nl.isygameclient.util.Vector2D; + +import java.io.IOException; +import java.lang.reflect.Array; +import java.lang.reflect.Type; +import java.nio.file.Files; +import java.nio.file.NoSuchFileException; +import java.nio.file.Paths; +import java.util.*; +import java.util.stream.Stream; + +public class Headless { + + private static final String JSON_FILENAME = "heuristics.json"; + + public static void main(String[] args) { + var heuristics = loadHeuristics(); + if (heuristics == null) { + System.err.println("No heuristics in file."); + return; + } + + Map<String, Game> games = new HashMap<>(); + + // Create Matches + for (Map.Entry<String, int[][]> heuristic1 : heuristics.entrySet()) { + for (Map.Entry<String, int[][]> heuristic2 : heuristics.entrySet()) { + + ArrayList<Player> players = new ArrayList<>(); + players.add(new Ai("ai1{ " + heuristic1.getKey() + "}", "black", heuristic1.getValue())); + players.add(new Ai("ai2{ " + heuristic2.getKey() + "}", "white", heuristic2.getValue())); + + var playerManager = new PlayerManager(0, players); + Othello othello = new Othello(playerManager); + games.put("ai1: " + heuristic1.getKey() + "; against " + "ai2: " + heuristic2.getKey(), othello); + } + } + + for (Map.Entry<String, Game> game : games.entrySet()) { + // add threading. + simulate(game.getKey(), game.getValue()); + } + } + + private static void simulate(String name, Game game) { + var playerManager = game.getPlayerManager(); + Map<Player, ArrayList<Long>> playersTimePerMoves = new HashMap<>(); + Map<Player, LinkedList<Vector2D<Integer, Integer>>> movesMade = new HashMap<>(); + + System.out.println("Starting game: " + name); + while (!game.isGameOver()) { + var currentPlayer = playerManager.getCurrentPlayer(); + if (game.getValidMoves(currentPlayer).isEmpty()) { + playerManager.nextPlayer(); + continue; + } + + + final long startGameTime = System.currentTimeMillis(); + var move = currentPlayer.onPlayerTurn(); + game.move(currentPlayer, move); + final long endTime = System.currentTimeMillis(); + + movesMade.computeIfAbsent(currentPlayer, k -> new LinkedList<>()).add(move); + playersTimePerMoves.computeIfAbsent(currentPlayer, k -> new ArrayList<>()).add((endTime - startGameTime)); + } + + System.out.println("Is game draw: " + game.isDraw()); + System.out.println("Game winners: " + game.getWinners().toString()); + System.out.println("Moves made: " + movesMade); + long totalGameTime = 0; + for (Map.Entry<Player, ArrayList<Long>> playerTimePerMoves: playersTimePerMoves.entrySet()) { + var player = playerTimePerMoves.getKey(); + var moveTimes = playerTimePerMoves.getValue(); + System.out.println(player + "'s Time per moves: " + moveTimes); + var optional = moveTimes.stream().reduce(Long::sum); + if (optional.isPresent()) { + totalGameTime += optional.get(); + } + } + System.out.println("Total game time in milliseconds: " + totalGameTime); + System.out.println(); + } + + private static Map<String, int[][]> loadHeuristics() { + try { + Type mapType = new TypeToken<Map<String, int[][]>>() { + }.getType(); + String inFile = new String(Files.readAllBytes(Paths.get(JSON_FILENAME))); + return new Gson().fromJson(inFile, mapType); + } catch (NoSuchFileException e) { + System.err.println("NO HEURISTICS JSON HAS BEEN PROVIDED."); + e.printStackTrace(); + } catch (IOException e) { + e.printStackTrace(); + } + return null; + } +} diff --git a/src/main/java/nl/isygameclient/controllers/GameSeletorMenu/GameCardController.java b/src/main/java/nl/isygameclient/controllers/GameSeletorMenu/GameCardController.java @@ -1,46 +0,0 @@ -package nl.isygameclient.controllers.GameSeletorMenu; - -import javafx.fxml.FXML; -import javafx.scene.control.Label; -import javafx.scene.image.Image; -import javafx.scene.image.ImageView; -import javafx.scene.input.MouseEvent; -import nl.isygameclient.Application; -import nl.isygameclient.models.GameCard; - -public class GameCardController { - - private GameCard gameCard; - - @FXML - private ImageView gameImage; - - @FXML - private Label gameTitle; - - public void initializeCard(GameCard gameCard) { - this.gameCard = gameCard; - // Set Title - gameTitle.setText(gameCard.name); - loadImage(); - } - - public void loadImage() { - // Load Image - if (gameCard.imgSrc != null) { - var imageSrc = Application.class.getResource(gameCard.imgSrc); - if (imageSrc != null) { - gameImage.setImage(new Image(imageSrc.toExternalForm())); - } else { - System.err.printf("Image for: %s at %s, doen not exist\n", gameCard.name, gameCard.imgSrc); - } - } else { - System.err.printf("ImgSrc for: %s, not provided\n", gameCard.name); - } - } - - @FXML - protected void onCardMouseClick(MouseEvent event) { - GameSelectorMenuController.onGameClick(event, gameCard); - } -} diff --git a/src/main/java/nl/isygameclient/controllers/GameSeletorMenu/GameSelectorMenuController.java b/src/main/java/nl/isygameclient/controllers/GameSeletorMenu/GameSelectorMenuController.java @@ -1,110 +0,0 @@ -package nl.isygameclient.controllers.GameSeletorMenu; - -import javafx.collections.FXCollections; -import javafx.fxml.FXML; -import javafx.fxml.FXMLLoader; -import javafx.fxml.Initializable; -import javafx.scene.Node; -import javafx.scene.control.ListCell; -import javafx.scene.control.ListView; -import javafx.scene.control.TextField; -import javafx.scene.input.MouseEvent; -import javafx.scene.layout.FlowPane; -import javafx.scene.layout.VBox; -import nl.isygameclient.Application; -import nl.isygameclient.models.GameCard; -import nl.isygameclient.util.SettingsFileHandler; -import nl.isygameclient.util.StageHandler; - -import java.io.IOException; -import java.net.URL; -import java.util.List; -import java.util.ResourceBundle; -import java.util.stream.Collectors; - -public class GameSelectorMenuController implements Initializable { - - private final List<GameCard> gameCards = SettingsFileHandler.load().gameCards; - - @FXML - public TextField searchBox; - @FXML - public ListView<GameCard> gamesList; - @FXML - public VBox gameDetail; - - @FXML - public FlowPane gameContainer; - - public static void onGameClick(MouseEvent event, GameCard gameCard) { - var stageHandler = StageHandler.get(); - var stageName = "Game"; - if (event.getClickCount() == 2 && gameCard.viewSrc != null && !stageHandler.getStage(stageName).isShowing()) { - stageHandler.changeSceneOfStage(stageName, gameCard.viewSrc); - var stage = stageHandler.getStage(stageName); - stage.setTitle(gameCard.name); - stage.show(); - stageHandler.iconifyStage("GameSelectorMenu"); - } - } - - @Override - public void initialize(URL url, ResourceBundle resourceBundle) { - initializeGamesListView(); - initializeGamesListCells(); - initializeGameCards(); - } - - private void initializeGamesListView() { - gamesList.setItems(FXCollections.observableList(gameCards)); - searchBox.textProperty().addListener((observable, oldValue, newValue) -> { - var filteredList = FXCollections.observableList(filterGameCards(newValue)); - gamesList.setItems(filteredList); - }); - } - - private void initializeGamesListCells() { - gamesList.setCellFactory(lv -> { - ListCell<GameCard> listCell = new ListCell<>() { - @Override - protected void updateItem(GameCard item, boolean empty) { - super.updateItem(item, empty); - - if (empty || item == null || item.name == null) { - setText(null); - } else { - setText(item.name); - } - } - }; - listCell.setOnMouseClicked((event) -> { - GameCard gameCard = listCell.getItem(); - onGameClick(event, gameCard); - }); - return listCell; - }); - } - - private void initializeGameCards() { - try { - for (GameCard gameCard : gameCards) { - FXMLLoader fxmlLoader = new FXMLLoader(Application.class.getResource("views/GameSelectorMenu/GameCard.fxml")); - Node node = fxmlLoader.load(); - GameCardController controller = fxmlLoader.getController(); - controller.initializeCard(gameCard); - gameContainer.getChildren().add(node); - } - } catch (IOException e) { - e.printStackTrace(); - } - } - - private List<GameCard> filterGameCards(String value) { - return gameCards.stream().filter(gameCard -> gameCard.name.contains(value)).collect(Collectors.toList()); - } - - @FXML - protected void onClearSearchButtonClick() { - searchBox.clear(); - } -} diff --git a/src/main/java/nl/isygameclient/controllers/Games/GameController.java b/src/main/java/nl/isygameclient/controllers/Games/GameController.java @@ -1,14 +0,0 @@ -package nl.isygameclient.controllers.Games; - -import javafx.fxml.FXML; -import nl.isygameclient.util.StageHandler; - -public abstract class GameController { - - @FXML - protected void onExitButtonClick() { - var settingsHandler = StageHandler.get(); - settingsHandler.getStage("Game").hide(); - settingsHandler.focusStage("GameSelectorMenu"); - } -} diff --git a/src/main/java/nl/isygameclient/controllers/Games/TicTacToe/TicTacToeMainMenuController.java b/src/main/java/nl/isygameclient/controllers/Games/TicTacToe/TicTacToeMainMenuController.java @@ -1,18 +0,0 @@ -package nl.isygameclient.controllers.Games.TicTacToe; - -import javafx.fxml.FXML; -import nl.isygameclient.controllers.Games.GameController; -import nl.isygameclient.util.StageHandler; - -public class TicTacToeMainMenuController extends GameController { - @FXML - public void onSinglePlayerButtonClick() { - StageHandler.get().changeSceneOfStage("Game", "views/Games/TicTacToe/TicTacToeSinglePlayer.fxml"); - } - - @FXML - public void onMultiplayerButtonClick() { - StageHandler.get().changeSceneOfStage("Game","views/Games/TicTacToe/TicTacToeMultiPlayer.fxml"); - } - -} diff --git a/src/main/java/nl/isygameclient/controllers/Games/TicTacToe/TicTacToeMultiPlayerController.java b/src/main/java/nl/isygameclient/controllers/Games/TicTacToe/TicTacToeMultiPlayerController.java @@ -1,203 +0,0 @@ -package nl.isygameclient.controllers.Games.TicTacToe; - -import com.jfoenix.controls.JFXButton; -import com.jfoenix.controls.JFXComboBox; -import java.io.IOException; -import java.net.URL; -import java.util.ResourceBundle; - -import javafx.application.Platform; -import javafx.event.ActionEvent; -import javafx.fxml.FXML; -import javafx.fxml.Initializable; -import javafx.scene.control.Label; -import javafx.scene.control.TextField; -import javafx.scene.layout.GridPane; -import nl.isygameclient.controllers.Games.GameController; -import nl.isygameclient.models.Games.TicTacToe; -import nl.isygameclient.network.GameClient; -import nl.isygameclient.network.GameClientException; -import nl.isygameclient.network.GameType; -import nl.isygameclient.network.Match; -import nl.isygameclient.util.StageHandler; - -public class TicTacToeMultiPlayerController extends GameController implements Runnable, Initializable { - private final JFXButton[] boardButtons = new JFXButton[TicTacToe.BOARD_SIZE * TicTacToe.BOARD_SIZE]; - private final String[] players = { "X", "O" }; - - @FXML - private Label playingAgainstLabel; - @FXML - private TextField hostField; - @FXML - private TextField portField; - @FXML - private TextField nameField; - @FXML - private TextField opponentField; - @FXML - private JFXComboBox<String> playingAsCombo; - @FXML - private Label currentPlayer; - @FXML - private Label gameOverText; - @FXML - private GridPane grid; - - private GameClient client; - private Match match; - private String playerSelf = "X", playerOther = "O"; - private boolean running; - - private Thread gameThread; - - @Override - public void initialize(URL url, ResourceBundle resourceBundle) { - for (int i = 0; i < TicTacToe.BOARD_SIZE * TicTacToe.BOARD_SIZE; i++) { - JFXButton button = new JFXButton(); - button.setId(Integer.toString(i)); - button.setMinSize(200.0, 200.0); - var styleClass = button.getStyleClass(); - styleClass.add("ttt-button"); - styleClass.add("display-large"); - button.setOnAction((ActionEvent event) -> onMoveButtonClick(button)); - boardButtons[i] = button; - grid.add(button, i / TicTacToe.BOARD_SIZE, i % TicTacToe.BOARD_SIZE); - } - - playingAsCombo.getItems().setAll(players); - playingAsCombo.getSelectionModel().selectFirst(); - - disableBoardButtons(); - // currentPlayer.setText(); - } - - public void run() { - running = true; - try { - client = new GameClient(hostField.getText(), Integer.parseInt(portField.getText())); - client.login(nameField.getText()); - - if (opponentField.getText().length() == 0) - match = client.match(GameType.TICTACTOE); - else - match = client.match(GameType.TICTACTOE, opponentField.getText()); - - while (running) { - match.update(); - - if (!match.isStarted()) - continue; - - // game.setCurrentPlayer(match.isYourTurn() ? playerSelf : playerOther); - // game.move(Integer.parseInt(moveMap.get("move"))); - Platform.runLater(() -> { - playingAgainstLabel.setText(match.getOpponent()); - - if (match.isYourTurn()) { - currentPlayer.setText(playerSelf); - enableBoardButtons(); - } else { - currentPlayer.setText(playerOther); - disableBoardButtons(); - } - for (int i = 0; i < TicTacToe.BOARD_SIZE * TicTacToe.BOARD_SIZE; i++) { - switch (match.getMove(i)) { - case -1 -> boardButtons[i].setText(playerOther); - case 0 -> boardButtons[i].setText(""); - case 1 -> boardButtons[i].setText(playerSelf); - } - } - }); - - if (match.getOutcome() == null) - continue; - - running = false; - - Platform.runLater(() -> { - switch (match.getOutcome().type) { - case DRAW: - System.out.println("Draw!"); - gameOverText.setText("Draw!"); - gameOverText.setVisible(true); - break; - case LOSS: - System.out.printf("%s, Is the Winner!\n\n", playerOther); - gameOverText.setText(String.format("%s, is the Winner!\n\n", playerOther)); - gameOverText.setVisible(true); - break; - case WIN: - System.out.printf("%s, Is the Winner!\n\n", playerSelf); - gameOverText.setText(String.format("%s, is the Winner!\n\n", playerSelf)); - gameOverText.setVisible(true); - default: - } - }); - } - match.abort(); - client.close(); - } catch (IOException e) { - e.printStackTrace(); - } - } - - @FXML - private void onNewGameButtonClick() throws NumberFormatException, InterruptedException { - gameOverText.setVisible(false); - - if (gameThread != null && running) { - running = false; - gameThread.join(); - } - gameThread = new Thread(this); - gameThread.start(); - } - - private void onMoveButtonClick(JFXButton button) { - try { - // Move - int pos = Integer.parseInt(button.getId()); - // if (game.isMoveValid(pos)) - match.move(pos); - // currentPlayer.setText(playerOther); - - // updateBoard(); - // currentPlayer.setText(playerSelf); - } catch (Exception exc) { - throw new RuntimeException(exc); - } - } - - private void disableBoardButtons() { - for (JFXButton button : boardButtons) { - button.setDisable(true); - } - } - - private void enableBoardButtons() { - for (JFXButton button : boardButtons) { - button.setDisable(false); - } - } - - @FXML - protected void onPlayingAsComboSelect() { - System.out.printf("Now playing As: %s\n", playingAsCombo.getValue()); - switch (playingAsCombo.getValue()) { - case "X" -> { - playerSelf = "X"; - playerOther = "O"; - } - case "O" -> { - playerSelf = "O"; - playerOther = "X"; - } - } - } - - @FXML - protected void onMainMenuButtonClick() { - StageHandler.get().changeSceneOfStage("Game","views/Games/TicTacToe/TicTacToeMainMenu.fxml"); - } -} diff --git a/src/main/java/nl/isygameclient/controllers/Games/TicTacToe/TicTacToeSinglePlayerController.java b/src/main/java/nl/isygameclient/controllers/Games/TicTacToe/TicTacToeSinglePlayerController.java @@ -1,157 +0,0 @@ -package nl.isygameclient.controllers.Games.TicTacToe; - -import com.jfoenix.controls.JFXButton; -import com.jfoenix.controls.JFXComboBox; - -import java.util.ArrayList; -import javafx.event.ActionEvent; -import javafx.fxml.FXML; -import javafx.scene.control.Label; -import javafx.scene.layout.GridPane; -import nl.isygameclient.controllers.Games.GameController; -import nl.isygameclient.models.Games.TicTacToe; -import nl.isygameclient.util.StageHandler; - -public class TicTacToeSinglePlayerController extends GameController { - private final ArrayList<JFXButton> boardButtons = new ArrayList<>(); - - @FXML - public JFXComboBox<String> difficultyCombo; - @FXML - public JFXComboBox<String> playingAsCombo; - @FXML - public JFXComboBox<String> opponentCombo; - - @FXML - protected Label currentPlayer; - @FXML - public Label gameOverText; - - @FXML - protected GridPane grid; - - private TicTacToe ttt = new TicTacToe(); - - @FXML - protected void initialize() { - initializeBoard(); - initializeDifficultyCombo(); - initializePlayingAsCombo(); - initializeOpponentCombo(); - updateCurrentPlayerLabel(); - } - - private void initializeDifficultyCombo() { - difficultyCombo.getItems().setAll("Easy", "Medium", "Hard"); - difficultyCombo.getSelectionModel().selectFirst(); - } - - private void initializePlayingAsCombo() { - playingAsCombo.getItems().setAll(ttt.getPlayers()); - playingAsCombo.getSelectionModel().selectFirst(); - } - - private void initializeOpponentCombo() { - opponentCombo.getItems().setAll("Human", "AI"); - opponentCombo.getSelectionModel().selectFirst(); - } - - private void initializeBoard() { - for (int i = 0; i < TicTacToe.BOARD_SIZE; i++) { - for (int j = 0; j < TicTacToe.BOARD_SIZE; j++) { - JFXButton button = new JFXButton(); - button.setId(String.valueOf(i + j * TicTacToe.BOARD_SIZE)); - button.setMinSize(200.0, 200.0); - var styleClass = button.getStyleClass(); - styleClass.add("ttt-button"); - styleClass.add("display-large"); - button.setOnAction((ActionEvent event) -> onMoveButtonClick(button)); - boardButtons.add(button); - grid.add(button, i, j); - } - } - } - - private void updateCurrentPlayerLabel() { - currentPlayer.setText(ttt.getCurrentPlayer()); - } - - private void onMoveButtonClick(JFXButton button) { - // Move - int pos = Integer.parseInt(button.getId()); - if (ttt.isMoveValid(pos)) { - ttt.move(pos); - button.setText(ttt.getCurrentPlayer()); - ttt.nextPlayer(); - updateCurrentPlayerLabel(); - } - - // Game Over - if (ttt.isGameOver()) { - onGameOver(); - } - } - - // TODO Create Game-over Modal - private void onGameOver() { - disableBoardButtons(); - System.out.println("Game Over"); - if (ttt.isDraw()) { - System.out.println("Draw!"); - gameOverText.setText("Draw!"); - gameOverText.setVisible(true); - } else { - System.out.printf("%s, Is the Winner!\n\n", ttt.getWinner()); - gameOverText.setText(String.format("%s, is the Winner!\n\n", ttt.getWinner())); - gameOverText.setVisible(true); - } - } - - private void clearBoardButtons() { - for (JFXButton button : boardButtons) { - button.setText(""); - } - } - - private void disableBoardButtons() { - for (JFXButton button : boardButtons) { - button.setDisable(true); - } - } - - private void enableBoardButtons() { - for (JFXButton button : boardButtons) { - button.setDisable(false); - } - } - - @FXML - protected void onDifficultyComboSelect() { - System.out.printf("Difficulty Changed to: %s\n", difficultyCombo.getValue()); - } - - @FXML - protected void onPlayingAsComboSelect() { - System.out.printf("Now playing As: %s\n", playingAsCombo.getValue()); - } - - @FXML - protected void onOpponentComboSelect() { - System.out.printf("Opponent Changed to: %s\n", opponentCombo.getValue()); - } - - @FXML - protected void onNewGameButtonClick() { - // Make new Game - ttt = new TicTacToe(); - - clearBoardButtons(); - enableBoardButtons(); - gameOverText.setVisible(false); - } - - @FXML - protected void onMainMenuButtonClick() { - StageHandler.get().changeSceneOfStage("Game","views/Games/TicTacToe/TicTacToeMainMenu.fxml"); - } -} diff --git a/src/main/java/nl/isygameclient/controllers/game_selector/GameCardController.java b/src/main/java/nl/isygameclient/controllers/game_selector/GameCardController.java @@ -0,0 +1,46 @@ +package nl.isygameclient.controllers.game_selector; + +import javafx.fxml.FXML; +import javafx.scene.control.Label; +import javafx.scene.image.Image; +import javafx.scene.image.ImageView; +import javafx.scene.input.MouseEvent; +import nl.isygameclient.Application; +import nl.isygameclient.models.settings.GameCard; + +public class GameCardController { + + private GameCard gameCard; + + @FXML + private ImageView gameImage; + + @FXML + private Label gameTitle; + + public void initializeCard(GameCard gameCard) { + this.gameCard = gameCard; + // Set Title + gameTitle.setText(gameCard.getName()); + loadImage(); + } + + public void loadImage() { + // Load Image + if (gameCard.getImgSrc() != null) { + var imageSrc = Application.class.getResource(gameCard.getImgSrc()); + if (imageSrc != null) { + gameImage.setImage(new Image(imageSrc.toExternalForm())); + } else { + System.err.printf("Image for: %s at %s, doen not exist\n", gameCard.getName(), gameCard.getImgSrc()); + } + } else { + System.err.printf("ImgSrc for: %s, not provided\n", gameCard.getName()); + } + } + + @FXML + protected void onCardMouseClick(MouseEvent event) { + GameSelectorController.onGameClick(event, gameCard); + } +} diff --git a/src/main/java/nl/isygameclient/controllers/game_selector/GameSelectorController.java b/src/main/java/nl/isygameclient/controllers/game_selector/GameSelectorController.java @@ -0,0 +1,110 @@ +package nl.isygameclient.controllers.game_selector; + +import javafx.collections.FXCollections; +import javafx.fxml.FXML; +import javafx.fxml.FXMLLoader; +import javafx.fxml.Initializable; +import javafx.scene.Node; +import javafx.scene.control.ListCell; +import javafx.scene.control.ListView; +import javafx.scene.control.TextField; +import javafx.scene.input.MouseEvent; +import javafx.scene.layout.FlowPane; +import javafx.scene.layout.VBox; +import nl.isygameclient.Application; +import nl.isygameclient.models.settings.GameCard; +import nl.isygameclient.util.SettingsFileHandler; +import nl.isygameclient.util.StageHandler; + +import java.io.IOException; +import java.net.URL; +import java.util.List; +import java.util.ResourceBundle; +import java.util.stream.Collectors; + +public class GameSelectorController implements Initializable { + + private final List<GameCard> gameCards = SettingsFileHandler.load().gameCards; + + @FXML + public TextField searchBox; + @FXML + public ListView<GameCard> gamesList; + @FXML + public VBox gameDetail; + + @FXML + public FlowPane gameContainer; + + public static void onGameClick(MouseEvent event, GameCard gameCard) { + var stageHandler = StageHandler.get(); + var stageName = "Game"; + if (event.getClickCount() == 2 && gameCard.getViewSrc() != null && !stageHandler.getStage(stageName).isShowing()) { + stageHandler.changeSceneOfStage(stageName, gameCard.getViewSrc()); + var stage = stageHandler.getStage(stageName); + stage.setTitle(gameCard.getName()); + stage.show(); + stageHandler.iconifyStage("GameSelector"); + } + } + + @Override + public void initialize(URL url, ResourceBundle resourceBundle) { + initializeGamesListView(); + initializeGamesListCells(); + initializeGameCards(); + } + + private void initializeGamesListView() { + gamesList.setItems(FXCollections.observableList(gameCards)); + searchBox.textProperty().addListener((observable, oldValue, newValue) -> { + var filteredList = FXCollections.observableList(filterGameCards(newValue)); + gamesList.setItems(filteredList); + }); + } + + private void initializeGamesListCells() { + gamesList.setCellFactory(lv -> { + ListCell<GameCard> listCell = new ListCell<>() { + @Override + protected void updateItem(GameCard item, boolean empty) { + super.updateItem(item, empty); + + if (empty || item == null || item.getName() == null) { + setText(null); + } else { + setText(item.getName()); + } + } + }; + listCell.setOnMouseClicked((event) -> { + GameCard gameCard = listCell.getItem(); + onGameClick(event, gameCard); + }); + return listCell; + }); + } + + private void initializeGameCards() { + try { + for (GameCard gameCard : gameCards) { + FXMLLoader fxmlLoader = new FXMLLoader(Application.class.getResource("views/game_selector/GameCard.fxml")); + Node node = fxmlLoader.load(); + GameCardController controller = fxmlLoader.getController(); + controller.initializeCard(gameCard); + gameContainer.getChildren().add(node); + } + } catch (IOException e) { + e.printStackTrace(); + } + } + + private List<GameCard> filterGameCards(String value) { + return gameCards.stream().filter(gameCard -> gameCard.getName().contains(value)).collect(Collectors.toList()); + } + + @FXML + protected void onClearSearchButtonClick() { + searchBox.clear(); + } +} diff --git a/src/main/java/nl/isygameclient/controllers/games/GameController.java b/src/main/java/nl/isygameclient/controllers/games/GameController.java @@ -0,0 +1,17 @@ +package nl.isygameclient.controllers.games; + +import javafx.fxml.FXML; +import nl.isygameclient.models.Game; +import nl.isygameclient.util.StageHandler; + +public abstract class GameController { + + protected Game game; + + @FXML + protected void onExitButtonClick() { + var settingsHandler = StageHandler.get(); + settingsHandler.getStage("Game").hide(); + settingsHandler.focusStage("GameSelector"); + } +} diff --git a/src/main/java/nl/isygameclient/controllers/games/othello/OthelloSinglePlayerController.java b/src/main/java/nl/isygameclient/controllers/games/othello/OthelloSinglePlayerController.java @@ -0,0 +1,134 @@ +package nl.isygameclient.controllers.games.othello; + +import com.jfoenix.controls.JFXButton; +import javafx.event.ActionEvent; +import javafx.fxml.FXML; +import javafx.fxml.Initializable; +import javafx.scene.control.Label; +import javafx.scene.layout.GridPane; +import javafx.scene.paint.Color; +import javafx.scene.shape.Circle; +import nl.isygameclient.controllers.games.GameController; +import nl.isygameclient.models.Player; +import nl.isygameclient.models.PlayerManager; +import nl.isygameclient.models.games.othello.Othello; +import nl.isygameclient.util.Vector2D; + +import java.net.URL; +import java.util.*; +import java.util.stream.Stream; + +public class OthelloSinglePlayerController extends GameController implements Initializable { + + private JFXButton[][] boardButtons; + + @FXML + private GridPane boardGrid; + + @Override + public void initialize(URL url, ResourceBundle resourceBundle) { +// var players = List.of(new Player("player1", "white"), new Player("player2", "black")); +// var manager = new PlayerManager(0, new ArrayList<>(players)); +// game = new Othello(manager); + initializeBoard(); + updateButtons(); + } + + private void initializeBoard() { + var board = game.getBoard(); + boardButtons = new JFXButton[board.getHeight()][board.getWidth()]; + for (int i = 0; i < board.getHeight(); i++) { + for (int j = 0; j < board.getWidth(); j++) { + JFXButton button = new JFXButton(); + var index = i + "," + j; + button.setId(index); + button.setMinSize(60.0, 60.0); + button.setMaxSize(60.0, 60.0); + + var styleClass = button.getStyleClass(); + styleClass.add("othello-button"); + styleClass.add("headline-small"); + button.setOnAction((ActionEvent event) -> onMoveButtonClick(button)); + boardButtons[i][j] = button; + boardGrid.add(button, i, j); + } + } + } + + private void updateButtons() { + clearBoardButtons(); + var board = game.getBoard(); + for (int i = 0; i < board.getHeight(); i++) { + for (int j = 0; j < board.getWidth(); j++) { + var index = board.get(new Vector2D<>(j,i)); + if (index == null) continue; + if (Objects.equals( "white", index.getPlayingAs())) { + addStone(boardButtons[i][j], Color.WHITE); + } else if(Objects.equals( "black", index.getPlayingAs())) { + addStone(boardButtons[i][j], Color.BLACK); + } + } + } + } + + private void addStone(JFXButton button, Color color) { + var circle = new Circle(); + circle.setRadius(20); + circle.setFill(color); + button.setGraphic(circle); + } + + private void onMoveButtonClick(JFXButton button) { + // Move + var id = button.getId().split(","); + var index = Stream.of(id).mapToInt(Integer::parseInt).toArray(); + var manager = game.getPlayerManager(); + var currentPlayer = manager.getCurrentPlayer(); + if (game.getValidMoves(currentPlayer).isEmpty()) { + manager.nextPlayer(); + return; + } + if (game.isMoveValid(currentPlayer, new Vector2D<>(index[0], index[1]))) { + game.move(currentPlayer, new Vector2D<>(index[0], index[1])); + updateButtons(); + manager.nextPlayer(); + } + + if (game.isGameOver()) { + disableBoardButtons(); + } + } + + private void clearBoardButtons() { + for (JFXButton[] row : boardButtons) { + for (JFXButton button : row) { + button.setGraphic(null); + } + } + } + + private void disableBoardButtons() { + for (JFXButton[] row : boardButtons) { + for (JFXButton button : row) { + button.setDisable(true); + } + } + } + + private void enableBoardButtons() { + for (JFXButton[] row : boardButtons) { + for (JFXButton button : row) { + button.setDisable(false); + } + } + } + + public void onNewGameButtonClick(ActionEvent actionEvent) { + game.restart(); + updateButtons(); + enableBoardButtons(); + } + public void onMainMenuButtonClick(ActionEvent actionEvent) { + } + +} diff --git a/src/main/java/nl/isygameclient/controllers/games/tictactoe/TicTacToeMainMenuController.java b/src/main/java/nl/isygameclient/controllers/games/tictactoe/TicTacToeMainMenuController.java @@ -0,0 +1,18 @@ +package nl.isygameclient.controllers.games.tictactoe; + +import javafx.fxml.FXML; +import nl.isygameclient.controllers.games.GameController; +import nl.isygameclient.util.StageHandler; + +public class TicTacToeMainMenuController extends GameController { + @FXML + public void onSinglePlayerButtonClick() { + StageHandler.get().changeSceneOfStage("Game", "views/Games/TicTacToe/TicTacToeSinglePlayer.fxml"); + } + + @FXML + public void onMultiplayerButtonClick() { + StageHandler.get().changeSceneOfStage("Game","views/Games/TicTacToe/TicTacToeMultiPlayer.fxml"); + } + +} diff --git a/src/main/java/nl/isygameclient/controllers/games/tictactoe/TicTacToeMultiPlayerController.java b/src/main/java/nl/isygameclient/controllers/games/tictactoe/TicTacToeMultiPlayerController.java @@ -0,0 +1,203 @@ +package nl.isygameclient.controllers.games.tictactoe; + +import com.jfoenix.controls.JFXButton; +import com.jfoenix.controls.JFXComboBox; +import java.io.IOException; +import java.net.URL; +import java.util.ArrayList; +import java.util.ResourceBundle; + +import javafx.application.Platform; +import javafx.event.ActionEvent; +import javafx.fxml.FXML; +import javafx.fxml.Initializable; +import javafx.scene.control.Label; +import javafx.scene.control.TextField; +import javafx.scene.layout.GridPane; +import nl.isygameclient.controllers.games.GameController; +import nl.isygameclient.models.games.tictactoe.TicTacToe; +import nl.isygameclient.network.GameClient; +import nl.isygameclient.network.GameType; +import nl.isygameclient.network.Match; +import nl.isygameclient.util.StageHandler; + +public class TicTacToeMultiPlayerController extends GameController implements Runnable, Initializable { + private final ArrayList<JFXButton> boardButtons = new ArrayList<>(); + private final String[] players = { "X", "O" }; + + @FXML + private Label playingAgainstLabel; + @FXML + private TextField hostField; + @FXML + private TextField portField; + @FXML + private TextField nameField; + @FXML + private TextField opponentField; + @FXML + private JFXComboBox<String> playingAsCombo; + @FXML + private Label currentPlayer; + @FXML + private Label gameOverText; + @FXML + private GridPane grid; + + private GameClient client; + private Match match; + private String playerSelf = "X", playerOther = "O"; + private boolean running; + + private Thread gameThread; + + @Override + public void initialize(URL url, ResourceBundle resourceBundle) { + for (int i = 0; i < 3 * 3; i++) { + JFXButton button = new JFXButton(); + button.setId(Integer.toString(i)); + button.setMinSize(200.0, 200.0); + var styleClass = button.getStyleClass(); + styleClass.add("ttt-button"); + styleClass.add("display-large"); + button.setOnAction((ActionEvent event) -> onMoveButtonClick(button)); + boardButtons.set(i, button); + grid.add(button, i / 3, i % 3); + } + + playingAsCombo.getItems().setAll(players); + playingAsCombo.getSelectionModel().selectFirst(); + + disableBoardButtons(); + // currentPlayer.setText(); + } + + public void run() { + running = true; + try { + client = new GameClient(hostField.getText(), Integer.parseInt(portField.getText())); + client.login(nameField.getText()); + + if (opponentField.getText().length() == 0) + match = client.match(GameType.TICTACTOE); + else + match = client.match(GameType.TICTACTOE, opponentField.getText()); + + while (running) { + match.update(); + + if (!match.isStarted()) + continue; + + // game.setCurrentPlayer(match.isYourTurn() ? playerSelf : playerOther); + // game.move(Integer.parseInt(moveMap.get("move"))); + Platform.runLater(() -> { + playingAgainstLabel.setText(match.getOpponent()); + + if (match.isYourTurn()) { + currentPlayer.setText(playerSelf); + enableBoardButtons(); + } else { + currentPlayer.setText(playerOther); + disableBoardButtons(); + } + for (int i = 0; i < 3 * 3; i++) { + switch (match.getMove(i)) { + case -1 -> boardButtons.get(i).setText(playerOther); + case 0 -> boardButtons.get(i).setText(""); + case 1 -> boardButtons.get(i).setText(playerSelf); + } + } + }); + + if (match.getOutcome() == null) + continue; + + running = false; + + Platform.runLater(() -> { + switch (match.getOutcome().type) { + case DRAW: + System.out.println("Draw!"); + gameOverText.setText("Draw!"); + gameOverText.setVisible(true); + break; + case LOSS: + System.out.printf("%s, Is the Winner!\n\n", playerOther); + gameOverText.setText(String.format("%s, is the Winner!\n\n", playerOther)); + gameOverText.setVisible(true); + break; + case WIN: + System.out.printf("%s, Is the Winner!\n\n", playerSelf); + gameOverText.setText(String.format("%s, is the Winner!\n\n", playerSelf)); + gameOverText.setVisible(true); + default: + } + }); + } + match.abort(); + client.close(); + } catch (IOException e) { + e.printStackTrace(); + } + } + + @FXML + private void onNewGameButtonClick() throws NumberFormatException, InterruptedException { + gameOverText.setVisible(false); + + if (gameThread != null && running) { + running = false; + gameThread.join(); + } + gameThread = new Thread(this); + gameThread.start(); + } + + private void onMoveButtonClick(JFXButton button) { + try { + // Move + int pos = Integer.parseInt(button.getId()); + // if (game.isMoveValid(pos)) + match.move(pos); + // currentPlayer.setText(playerOther); + + // updateBoard(); + // currentPlayer.setText(playerSelf); + } catch (Exception exc) { + throw new RuntimeException(exc); + } + } + + private void disableBoardButtons() { + for (JFXButton button : boardButtons) { + button.setDisable(true); + } + } + + private void enableBoardButtons() { + for (JFXButton button : boardButtons) { + button.setDisable(false); + } + } + + @FXML + protected void onPlayingAsComboSelect() { + System.out.printf("Now playing As: %s\n", playingAsCombo.getValue()); + switch (playingAsCombo.getValue()) { + case "X" -> { + playerSelf = "X"; + playerOther = "O"; + } + case "O" -> { + playerSelf = "O"; + playerOther = "X"; + } + } + } + + @FXML + protected void onMainMenuButtonClick() { + StageHandler.get().changeSceneOfStage("Game","views/Games/TicTacToe/TicTacToeMainMenu.fxml"); + } +} diff --git a/src/main/java/nl/isygameclient/controllers/games/tictactoe/TicTacToeSinglePlayerController.java b/src/main/java/nl/isygameclient/controllers/games/tictactoe/TicTacToeSinglePlayerController.java @@ -0,0 +1,144 @@ +package nl.isygameclient.controllers.games.tictactoe; + +import com.jfoenix.controls.JFXButton; +import com.jfoenix.controls.JFXComboBox; + +import java.util.ArrayList; +import java.util.List; +import java.util.stream.Stream; + +import javafx.event.ActionEvent; +import javafx.fxml.FXML; +import javafx.scene.control.Label; +import javafx.scene.layout.GridPane; +import nl.isygameclient.controllers.games.GameController; +import nl.isygameclient.models.Player; +import nl.isygameclient.models.PlayerManager; +import nl.isygameclient.models.games.tictactoe.TicTacToe; +import nl.isygameclient.util.StageHandler; +import nl.isygameclient.util.Vector2D; + +public class TicTacToeSinglePlayerController extends GameController { + private final ArrayList<JFXButton> boardButtons = new ArrayList<>(); + + @FXML + public JFXComboBox<String> difficultyCombo; + @FXML + public JFXComboBox<String> playingAsCombo; + @FXML + public JFXComboBox<String> opponentCombo; + + @FXML + protected Label currentPlayer; + @FXML + public Label gameOverText; + + @FXML + protected GridPane grid; + + private Thread gameThread; + + @FXML + protected void initialize() { + initializeGame(); + initializeBoard(); + updateCurrentPlayerLabel(); + } + + private void initializeGame() { +// var players = new ArrayList<>(List.of(new Player("player1", "x"), new Player("player2", "o"))); +// var manager = new PlayerManager(0, players); +// game = new TicTacToe(manager); + + } + + + private void initializeBoard() { + var board = game.getBoard(); + for (int y = 0; y < board.getWidth(); y++) { + for (int x = 0; x < board.getHeight(); x++) { + JFXButton button = new JFXButton(); + + var index = x + "," + y; + button.setId(index); + button.setMinSize(200.0, 200.0); + var styleClass = button.getStyleClass(); + styleClass.add("ttt-button"); + styleClass.add("display-large"); + button.setOnAction((ActionEvent event) -> onMoveButtonClick(button)); + boardButtons.add(button); + grid.add(button, x, y); + } + } + } + + private void updateCurrentPlayerLabel() { + currentPlayer.setText(game.getPlayerManager().getCurrentPlayer().getPlayingAs()); + } + + private void onMoveButtonClick(JFXButton button) { + // Move + var id = button.getId().split(","); + var index = Stream.of(id).mapToInt(Integer::parseInt).toArray(); + var manager = game.getPlayerManager(); + var currentPlayer = manager.getCurrentPlayer(); + if (game.isMoveValid(currentPlayer, new Vector2D<>(index[0], index[1]))) { + game.move(currentPlayer, new Vector2D<>(index[0], index[1])); + button.setText(currentPlayer.getPlayingAs()); + updateCurrentPlayerLabel(); + manager.nextPlayer(); + } + + // Game Over + if (game.isGameOver()) { + onGameOver(); + } + } + + private void onGameOver() { + disableBoardButtons(); + System.out.println("Game Over"); + if (game.isDraw()) { + System.out.println("Draw!"); + gameOverText.setText("Draw!"); + gameOverText.setVisible(true); + } else { + System.out.printf("%s, Is the Winner!\n\n", game.getWinners().get(0)); + gameOverText.setText(String.format("%s, is the Winner!\n\n", game.getWinners().get(0))); + gameOverText.setVisible(true); + } + } + + private void clearBoardButtons() { + for (JFXButton button : boardButtons) { + button.setText(""); + } + } + + private void disableBoardButtons() { + for (JFXButton button : boardButtons) { + button.setDisable(true); + } + } + + private void enableBoardButtons() { + for (JFXButton button : boardButtons) { + button.setDisable(false); + } + } + + @FXML + protected void onNewGameButtonClick() { + // Make new Game + game.restart(); + + clearBoardButtons(); + enableBoardButtons(); + gameOverText.setVisible(false); + } + + @FXML + protected void onMainMenuButtonClick() { + StageHandler.get().changeSceneOfStage("Game","views/Games/TicTacToe/TicTacToeMainMenu.fxml"); + } +} diff --git a/src/main/java/nl/isygameclient/models/Ai.java b/src/main/java/nl/isygameclient/models/Ai.java @@ -0,0 +1,133 @@ +package nl.isygameclient.models; + +import nl.isygameclient.util.Vector2D; + +import java.util.*; + +public class Ai extends Player{ + + public static final int DEPTH = 5; + + private int[][] heuristics; + + public Ai(String name, String playingAs, int[][] heuristics) { + super(name, playingAs); + this.heuristics = heuristics; + } + + public static <T> T randomMove(Collection<T> possibleMoves) { + int size = possibleMoves.size(); + int item = new Random().nextInt(size); // In real life, the Random object should be rather more shared than this + int i = 0; + for (T obj : possibleMoves) + if (i == item) + return obj; + else + i++; + return null; + } + + private double miniMax(Integer depth, boolean isMaximizing) { + var playerManager = game.getPlayerManager(); + var currentPlayer = playerManager.getCurrentPlayer(); + var possibleMoves = game.getValidMoves(currentPlayer); + + if (depth == 0 || game.isGameOver()) { + return heuristicScore(currentPlayer); + } + + double bestValue = isMaximizing ? 0 : Double.MAX_VALUE; + for (Vector2D<Integer, Integer> move : possibleMoves) { + game.move(currentPlayer, move); + double value = miniMax(depth - 1, !isMaximizing); + bestValue = isMaximizing ? Math.max(bestValue, value) : Math.min(bestValue, value); + game.undo(); + } + return bestValue; + } + + private double miniMaxAlphaBeta(double alpha, double beta, Integer depth, boolean isMaximizing) { + var playerManager = game.getPlayerManager(); + var currentPlayer = playerManager.getCurrentPlayer(); + var possibleMoves = game.getValidMoves(currentPlayer); + + if (depth == 0 || game.isGameOver()) { + return heuristicScore(currentPlayer); + } + + double bestValue; + if (isMaximizing) { + bestValue = Double.MIN_VALUE; + for (Vector2D<Integer, Integer> move : possibleMoves) { + game.move(currentPlayer, move); + double value = miniMaxAlphaBeta(alpha, beta, depth - 1, false); + game.undo(); + + bestValue = Math.max(bestValue, value); + alpha = Math.max(alpha, bestValue); + if (beta <= alpha) { + break; + } + } + + } else { + bestValue = Double.MAX_VALUE; + for (Vector2D<Integer, Integer> move : possibleMoves) { + game.move(currentPlayer, move); + double value = miniMaxAlphaBeta(alpha, beta, depth - 1, true); + game.undo(); + + bestValue = Math.min(bestValue, value); + beta = Math.min(beta, bestValue); + if (beta <= alpha) { + break; + } + } + } + return bestValue; + } + + @Override + public Vector2D<Integer, Integer> onPlayerTurn() { + var manager = game.getPlayerManager(); + var currentPlayer = manager.getCurrentPlayer(); + var possibleMoves = game.getValidMoves(currentPlayer); + + double bestValue = Integer.MIN_VALUE; + var bestMove = randomMove(possibleMoves); + + for (Vector2D<Integer, Integer> move : possibleMoves) { + game.move(currentPlayer, move); + double moveValue = miniMaxAlphaBeta(Double.MIN_VALUE, Double.MAX_VALUE, DEPTH, false); + if (moveValue > bestValue) { + bestValue = moveValue; + bestMove = move; + } + game.undo(); + } + return bestMove; + } + + public int heuristicScore(Player player) { + var board = game.getBoard(); + int score = 0; + for (int y = 0; y < board.getHeight(); y++) { + for (int x = 0; x < board.getWidth(); x++) { + if (Objects.equals(player, board.get(new Vector2D<>(x, y)))) { + score += heuristics[y][x]; + } + } + } + return score; + } + + public void setGame(Game game) { + this.game = game; + } + + public void setHeuristics(int[][] heuristics) { + this.heuristics = heuristics; + } + + +} diff --git a/src/main/java/nl/isygameclient/models/Game.java b/src/main/java/nl/isygameclient/models/Game.java @@ -0,0 +1,54 @@ +package nl.isygameclient.models; + + +import nl.isygameclient.models.board.Board; +import nl.isygameclient.models.board.HistoryBoard; +import nl.isygameclient.util.Vector2D; + +import java.util.List; + +public abstract class Game { + + protected PlayerManager playerManager; + protected HistoryBoard<Player> board; + + public Game(PlayerManager playerManager, HistoryBoard<Player> board) { + this.playerManager = playerManager; + this.board = board; + + for (Player player : playerManager.getPlayers()) { + player.setGame(this); + } + } + + public abstract void restart(); + + public abstract boolean move(Player player, Vector2D<Integer, Integer> pos); + + public abstract boolean isMoveValid(Player player, Vector2D<Integer, Integer> pos); + + public abstract boolean isDraw(); + + public abstract boolean isGameOver(); + + public abstract boolean isWinner(Player player); + + public void undo() { + board.undo(); + playerManager.previousPlayer(); + } + + public abstract List<Vector2D<Integer, Integer>> getValidMoves(Player player); + + public abstract List<Player> getWinners(); + + public abstract int getPlayerScore(Player player); + + public PlayerManager getPlayerManager() { + return playerManager; + } + + public HistoryBoard<Player> getBoard() { + return board; + } +} diff --git a/src/main/java/nl/isygameclient/models/GameCard.java b/src/main/java/nl/isygameclient/models/GameCard.java @@ -1,13 +0,0 @@ -package nl.isygameclient.models; - -public class GameCard { - public String name; - public String viewSrc; - public String imgSrc; - - public GameCard(String name, String viewSrc, String imgSrc) { - this.name = name; - this.viewSrc = viewSrc; - this.imgSrc = imgSrc; - } -} diff --git a/src/main/java/nl/isygameclient/models/Games/Difficulty.java b/src/main/java/nl/isygameclient/models/Games/Difficulty.java @@ -1,13 +0,0 @@ -package nl.isygameclient.models.Games; - -public enum Difficulty { - EASY("Easy"), - MEDIUM("Medium"), - HARD("Hard"); - - public final String name; - - private Difficulty(String name_) { - name = name_; - } -} diff --git a/src/main/java/nl/isygameclient/models/Games/Game.java b/src/main/java/nl/isygameclient/models/Games/Game.java @@ -1,52 +0,0 @@ -package nl.isygameclient.models.Games; - -import java.util.Arrays; -import java.util.List; - - -public abstract class Game { - int currentPlayerIndex; - String[] players; - - public Game(int currentPlayer, String[] players) { - this.currentPlayerIndex = currentPlayer; - this.players = players; - } - - public abstract void move(int pos); - - public abstract boolean isMoveValid(int pos); - - public abstract List<Integer> getPossibleMoves(); - - public abstract boolean isGameOver(); - - public abstract String getWinner(); - - public void nextPlayer() { - currentPlayerIndex += 1; - if (currentPlayerIndex >= players.length) { - currentPlayerIndex = 0; - } - } - - public String getCurrentPlayer() { - return players[currentPlayerIndex]; - } - - public void setCurrentPlayerIndex(int playerIndex) { - this.currentPlayerIndex = playerIndex; - } - - public void setCurrentPlayer(String player) { - currentPlayerIndex = Arrays.asList(players).indexOf(player); - } - - public int getCurrentPlayerIndex() { - return currentPlayerIndex; - } - - public String[] getPlayers() { - return players; - } -} diff --git a/src/main/java/nl/isygameclient/models/Games/TicTacToe.java b/src/main/java/nl/isygameclient/models/Games/TicTacToe.java @@ -1,74 +0,0 @@ -package nl.isygameclient.models.Games; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.List; - -public class TicTacToe extends Game { - public static final int BOARD_SIZE = 3; - - String[] board = new String[BOARD_SIZE * BOARD_SIZE]; - - public TicTacToe() { - super(0, new String[] { "X", "O" }); - } - - public boolean isMoveValid(int pos) { - return (pos >= 0 && pos < (BOARD_SIZE * BOARD_SIZE)) && board[pos] == null; - } - - public void move(int pos) { - board[pos] = players[currentPlayerIndex]; - } - - public List<Integer> getPossibleMoves() { - List<Integer> possibleMoves = new ArrayList<>(); - for (int i = 0; i < board.length; i++) { - if (isMoveValid(i)) possibleMoves.add(i); - } - return possibleMoves; - } - - public boolean isDraw() { - return !Arrays.asList(board).contains(null); - } - - public boolean isGameOver() { - return getWinner() != null || isDraw(); - } - - public String getWinner() { - for (String player : players) { - boolean topRow = player.equals(board[0]) && player.equals(board[1]) && player.equals(board[2]); - boolean midRow = player.equals(board[3]) && player.equals(board[4]) && player.equals(board[5]); - boolean botRow = player.equals(board[6]) && player.equals(board[7]) && player.equals(board[8]); - - boolean leftCol = player.equals(board[0]) && player.equals(board[3]) && player.equals(board[6]); - boolean midCol = player.equals(board[1]) && player.equals(board[4]) && player.equals(board[7]); - boolean rightCol = player.equals(board[2]) && player.equals(board[5]) && player.equals(board[8]); - - boolean lrCross = player.equals(board[0]) && player.equals(board[4]) && player.equals(board[8]); - boolean rlCross = player.equals(board[2]) && player.equals(board[4]) && player.equals(board[6]); - - if (topRow || midRow || botRow || leftCol || midCol || rightCol || lrCross || rlCross) { - return player; - } - } - return null; - } - - public TicTacToe clone() throws CloneNotSupportedException { - TicTacToe cloneBoard = (TicTacToe) super.clone(); - cloneBoard.board = board.clone(); - return cloneBoard; - } - - public void printGameBoard() { - for (int i = 0; i < board.length; i++) { - System.out.print(board[i] == null ? " " : board[i]); - if (i % BOARD_SIZE < BOARD_SIZE - 1) System.out.print("|"); - if (i % BOARD_SIZE == BOARD_SIZE - 1 && i < board.length - 1) System.out.println("\n-+-+-"); - } - System.out.println(); - } -} -\ No newline at end of file diff --git a/src/main/java/nl/isygameclient/models/Player.java b/src/main/java/nl/isygameclient/models/Player.java @@ -0,0 +1,49 @@ +package nl.isygameclient.models; + +import nl.isygameclient.util.Vector2D; + +public abstract class Player { + + protected Game game; + protected String playerName; + protected String playingAs; + + public Player(String playerName, String playingAs) { + this.playerName = playerName; + this.playingAs = playingAs; + } + + public abstract Vector2D<Integer, Integer> onPlayerTurn(); + + public Game getGame() { + return game; + } + + public void setGame(Game game) { + this.game = game; + } + + public String getPlayerName() { + return playerName; + } + + public void setPlayerName(String playerName) { + this.playerName = playerName; + } + + public String getPlayingAs() { + return playingAs; + } + + public void setPlayingAs(String playingAs) { + this.playingAs = playingAs; + } + + @Override + public String toString() { + return "Player{" + + "playerName='" + playerName + '\'' + + ", playingAs='" + playingAs + '\'' + + '}'; + } +} diff --git a/src/main/java/nl/isygameclient/models/PlayerManager.java b/src/main/java/nl/isygameclient/models/PlayerManager.java @@ -0,0 +1,67 @@ +package nl.isygameclient.models; + +import java.util.ArrayList; +import java.util.Objects; + +public class PlayerManager { + protected int startingPlayerIndex; + protected int currentPlayerIndex; + protected final ArrayList<Player> players; + + public PlayerManager(int startingPlayerIndex, ArrayList<Player> players) { + this.startingPlayerIndex = startingPlayerIndex; + this.currentPlayerIndex = startingPlayerIndex; + this.players = players; + } + + public void restart() { + currentPlayerIndex = startingPlayerIndex; + } + + public void nextPlayer() { + currentPlayerIndex += 1; + if (currentPlayerIndex >= players.size()) { + currentPlayerIndex = 0; + } + } + public void previousPlayer() { + currentPlayerIndex -= 1; + if (currentPlayerIndex < 0) { + currentPlayerIndex = players.size() - 1; + } + } + + public boolean isCurrentPlayer(Player player) { + return Objects.equals(player, getCurrentPlayer()); + } + + public Player getCurrentPlayer(){ + return players.get(currentPlayerIndex); + } + + public Player getStartingPlayer() { + return players.get(startingPlayerIndex); + } + + public ArrayList<Player> getPlayers(){ + return players; + } + + public int getStartingPlayerIndex() { + return startingPlayerIndex; + } + + public int getCurrentPlayerIndex() { + return currentPlayerIndex; + } + + public void setStartingPlayerIndex(int startingPlayerIndex) { + this.startingPlayerIndex = startingPlayerIndex; + } + + public void setCurrentPlayerIndex(int currentPlayerIndex) { + this.currentPlayerIndex = currentPlayerIndex; + } + + +} diff --git a/src/main/java/nl/isygameclient/models/Settings/Settings.java b/src/main/java/nl/isygameclient/models/Settings/Settings.java @@ -1,29 +0,0 @@ -package nl.isygameclient.models.Settings; - - -import nl.isygameclient.models.GameCard; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.Map; - - -/** - * This Class holds default settings for the application. - * It also provides the format of the settings that are saved to a file. - * - * @author A F Koens - */ -public class Settings { - public Map<String, StageSettings> stages = new HashMap<>(Map.of( - "GameSelectorMenu", new StageSettings())); - - public boolean tournamentMode = false; - - public ArrayList<GameCard> gameCards = new ArrayList<>(Arrays.asList( - new GameCard("TicTacToe", "views/Games/TicTacToe/TicTacToeMainMenu.fxml", "images/tictactoe_logo.png"), - new GameCard("Othello", null, "images/othello_logo.png"), - new GameCard("Starvation", null, null)) - ); -} diff --git a/src/main/java/nl/isygameclient/models/Settings/StageSettings.java b/src/main/java/nl/isygameclient/models/Settings/StageSettings.java @@ -1,61 +0,0 @@ -package nl.isygameclient.models.Settings; - -import javafx.stage.Stage; - -public class StageSettings { - public String Title = "ISYGameClient"; - public double x = 0; - public double y = 0; - public double width = 680; - public double height = 480; - public double minWidth = 0; - public double minHeight = 0; - public double maxWidth = 1.7976931348623157E308; - public double maxHeight = 1.7976931348623157E308; - public boolean isIconified = false; - public boolean isAlwaysOnTop = false; - public boolean isFullScreen = true; - public boolean isResizable = true; - public boolean isMaximized = false; - public boolean isShowing = false; - - - public static StageSettings createSettingsFromStage(Stage stage) { - var stageSettings = new StageSettings(); - stageSettings.Title = stage.getTitle(); - stageSettings.x = stage.getX(); - stageSettings.y = stage.getY(); - stageSettings.width = stage.getWidth(); - stageSettings.height = stage.getHeight(); - stageSettings.minWidth = stage.getMinWidth(); - stageSettings.minHeight = stage.getMinHeight(); - stageSettings.maxWidth = stage.getMaxWidth(); - stageSettings.maxHeight = stage.getMaxHeight(); - stageSettings.isIconified = stage.isIconified(); - stageSettings.isAlwaysOnTop = stage.isAlwaysOnTop(); - stageSettings.isFullScreen = stage.isFullScreen(); - stageSettings.isResizable = stage.isResizable(); - stageSettings.isMaximized = stage.isMaximized(); - stageSettings.isShowing = stage.isShowing(); - return stageSettings; - } - - public static Stage configureStageFromSettings(StageSettings stageSettings) { - Stage stage = new Stage(); - stage.setTitle(stageSettings.Title); - stage.setX(stageSettings.x); - stage.setY(stageSettings.y); - stage.setWidth(stageSettings.width); - stage.setHeight(stageSettings.height); - stage.setMinWidth(stageSettings.minWidth); - stage.setMinHeight(stageSettings.minHeight); - stage.setMaxWidth(stageSettings.maxWidth); - stage.setMaxHeight(stageSettings.maxHeight); - stage.setIconified(stageSettings.isIconified); - stage.setAlwaysOnTop(stageSettings.isAlwaysOnTop); - stage.setFullScreen(stageSettings.isFullScreen); - stage.setResizable(stageSettings.isResizable); - stage.setMaximized(stageSettings.isMaximized); - return stage; - } -} diff --git a/src/main/java/nl/isygameclient/models/board/Board.java b/src/main/java/nl/isygameclient/models/board/Board.java @@ -0,0 +1,65 @@ +package nl.isygameclient.models.board; + +import nl.isygameclient.models.Player; +import nl.isygameclient.util.Vector2D; + +import java.util.Arrays; +import java.util.List; + +public abstract class Board<T> { + + protected final int width; + protected final int height; + + protected final T[][] field; + + @SuppressWarnings("unchecked") + public Board(int width, int height) { + this.width = width; + this.height = height; + this.field = (T[][]) new Player[height][width]; + } + + public void set(T value, Vector2D<Integer, Integer> pos) { + field[pos.getY()][pos.getX()] = value; + } + + public T get(Vector2D<Integer, Integer> pos) { + return field[pos.getY()][pos.getX()]; + } + + public int size() { + return width * height; + } + + public void clear() { + Arrays.fill(field, null); + } + + public int getWidth() { + return width; + } + + public int getHeight() { + return height; + } + + public T[][] getField() { + return field; + } + + @Override + public String toString() { + StringBuilder builder = new StringBuilder(); + for (T[] row : field) { + builder.append(Arrays.toString(row)) + .append(", \n"); + } + + return "Board{" + + "width=" + width + + ", height=" + height + + ", field=\n" + builder + + '}'; + } +} diff --git a/src/main/java/nl/isygameclient/models/board/HistoryBoard.java b/src/main/java/nl/isygameclient/models/board/HistoryBoard.java @@ -0,0 +1,91 @@ +package nl.isygameclient.models.board; + + +import nl.isygameclient.util.Vector2D; + +import java.util.*; + +public abstract class HistoryBoard<T> extends Board<T> { + public record Move<T>(T current, T previous) { + } + + public record Change<T>(T player, Map<Vector2D<Integer, Integer>, Move<T>> moves) { + } + + private int index; + private final Stack<Change<T>> stack; + + public HistoryBoard(int width, int height) { + super(width, height); + this.stack = new Stack<>(); + this.index = 0; + } + + public List<Change<T>> getHistory() { + return stack.subList(0, index); + } + + public List<Change<T>> getFuture() { + return stack.subList(index, stack.size()); + } + + public void add(T player, Map<Vector2D<Integer, Integer>, T> changes) { + for (int i = stack.size(); i > index; i--) + stack.pop(); + + Map<Vector2D<Integer, Integer>, Move<T>> moves = new HashMap<>(); + for (Map.Entry<Vector2D<Integer, Integer>, T> entry : changes.entrySet()) { + var vector = entry.getKey(); + moves.put(entry.getKey(), new Move<>(entry.getValue(), get(vector))); + } + + stack.push(new Change<>(player, moves)); + index++; + + for (var move : changes.entrySet()) { + var vector = move.getKey(); + set(move.getValue(), vector); + } + } + + public boolean canUndo() { + return index > 0; + } + + public boolean canRedo() { + return index < stack.size(); + } + + public boolean undo() { + if (!canUndo()) + return false; + + index--; + + for (var move : stack.get(index).moves.entrySet()) { + var vector = move.getKey(); + set(move.getValue().previous, vector); + } + + return true; + } + + public boolean redo() { + if (!canRedo()) + return false; + + for (var move : stack.get(index).moves.entrySet()) { + var vector = move.getKey(); + set(move.getValue().current, vector); + } + + index++; + return true; + } + + public void clear() { + super.clear(); + stack.clear(); + index = 0; + } +} diff --git a/src/main/java/nl/isygameclient/models/games/othello/Othello.java b/src/main/java/nl/isygameclient/models/games/othello/Othello.java @@ -0,0 +1,178 @@ +package nl.isygameclient.models.games.othello; + +import nl.isygameclient.models.Game; +import nl.isygameclient.models.Player; +import nl.isygameclient.models.PlayerManager; +import nl.isygameclient.models.board.HistoryBoard; +import nl.isygameclient.util.Vector2D; + +import java.util.*; +import java.util.stream.Collectors; + +public class Othello extends Game { + + public Othello(PlayerManager playerManager) { + super(playerManager, new HistoryBoard<>(8, 8) { + }); + initializeBoard(); + } + + @Override + public void restart() { + playerManager.restart(); + board = new HistoryBoard<>(8, 8) { + }; + initializeBoard(); + } + + private void initializeBoard() { + var players = playerManager.getPlayers(); + board.set(players.get(0), new Vector2D<>(3, 3)); + board.set(players.get(0), new Vector2D<>(4, 4)); + board.set(players.get(1), new Vector2D<>(3, 4)); + board.set(players.get(1), new Vector2D<>(4, 3)); + + } + + @Override + public boolean isDraw() { + var scores = getScores(); + return scores.values().stream().distinct().count() == 1; + } + + @Override + public boolean isGameOver() { + for (Player player : playerManager.getPlayers()) { + if (getValidMoves(player).size() > 0){ + return false; + } + } + return true; + } + + @Override + public boolean isWinner(Player player) { + Player winner = null; + int score = 0; + for (Map.Entry<Player, Integer> entry : getScores().entrySet()) { + if (entry.getValue() > score) { + winner = entry.getKey(); + score = entry.getValue(); + } + } + return Objects.equals(player, winner); + } + + + @Override + public boolean move(Player player, Vector2D<Integer, Integer> pos) { + if (!isMoveValid(player, pos)) { + return false; + } + + var flippable = checkNeighbours(player, pos); + + Map<Vector2D<Integer, Integer>, Player> changes = flippable.stream().collect(Collectors.toMap(e -> e, e -> player)); + board.add(player, changes); + playerManager.nextPlayer(); + return true; + } + + + @Override + public boolean isMoveValid(Player player, Vector2D<Integer, Integer> pos) { + if (board.get(pos) != null) + return false; + + var flippable = checkNeighbours(player, pos); + return !flippable.isEmpty(); + } + + + private List<Vector2D<Integer, Integer>> rangeContains(Player player, Vector2D<Integer, Integer> pos, Vector2D<Integer, Integer> dir) { + List<Vector2D<Integer, Integer>> flippable = new ArrayList<>(); + int x = pos.getX(); + int y = pos.getY(); + while (x >= 0 && x < board.getWidth() && y >= 0 && y < board.getHeight()) { + var newPos = new Vector2D<>(x, y); + flippable.add(newPos); + if (Objects.equals(board.get(newPos), player)) { + return flippable; + } + x += dir.getX(); + y += dir.getY(); + } + return null; + } + + + private Set<Vector2D<Integer, Integer>> checkNeighbours(Player player, Vector2D<Integer, Integer> pos) { + Set<Vector2D<Integer, Integer>> flippable = new HashSet<>(); + for (int dx = -1; dx <= 1; dx++) { + if ((pos.getX() + dx) < 0 || (pos.getX() + dx) >= board.getWidth()) + continue; + + for (int dy = -1; dy <= 1; dy++) { + if ((pos.getY() + dy) < 0 || (pos.getY() + dy) >= board.getHeight()) + continue; + + if (dx == 0 && dy == 0) + continue; + + var checkPos = board.get(new Vector2D<>(pos.getX() + dx, pos.getY() + dy)); + if (checkPos != null && checkPos != player) { + List<Vector2D<Integer, Integer>> flip = rangeContains(player, pos, new Vector2D<>(dx, dy)); + if (flip != null) + flippable.addAll(flip); + } + } + } + return flippable; + } + + @Override + public List<Vector2D<Integer, Integer>> getValidMoves(Player player) { + List<Vector2D<Integer, Integer>> valid = new ArrayList<>(); + for (int y = 0; y < board.getHeight(); y++) { + for (int x = 0; x < board.getWidth(); x++) { + if (isMoveValid(player, new Vector2D<>(x, y))) { + valid.add(new Vector2D<>(x, y)); + } + } + } + return valid; + } + + @Override + public List<Player> getWinners() { + ArrayList<Player> winners = new ArrayList<>(); + for (Player player : playerManager.getPlayers()) { + if (isWinner(player)) { + winners.add(player); + } + } + return winners; + } + + @Override + public int getPlayerScore(Player player) { + int score = 0; + for (int y = 0; y < board.getHeight(); y++) { + for (int x = 0; x < board.getWidth(); x++) { + if (Objects.equals(player, board.get(new Vector2D<>(x, y)))) { + score += 1; + } + } + } + return score; + } + + private HashMap<Player, Integer> getScores() { + HashMap<Player, Integer> scores = new HashMap<>(); + for (Player player : playerManager.getPlayers()) { + int score = getPlayerScore(player); + scores.put(player, score); + } + return scores; + } +} diff --git a/src/main/java/nl/isygameclient/models/games/tictactoe/TicTacToe.java b/src/main/java/nl/isygameclient/models/games/tictactoe/TicTacToe.java @@ -0,0 +1,103 @@ +package nl.isygameclient.models.games.tictactoe; + +import nl.isygameclient.models.Game; +import nl.isygameclient.models.Player; +import nl.isygameclient.models.PlayerManager; +import nl.isygameclient.models.board.HistoryBoard; +import nl.isygameclient.util.Vector2D; + +import java.util.*; + +public class TicTacToe extends Game { + + public TicTacToe(PlayerManager playerManager) { + super(playerManager, new HistoryBoard<>(3,3){}); + } + + @Override + public void restart() { + playerManager.restart(); + board = new HistoryBoard<>(3,3){}; + } + + @Override + public boolean move(Player player, Vector2D<Integer, Integer> pos) { + if (!isMoveValid(player, pos)) { + return false; + } + + var currentPlayer = playerManager.getCurrentPlayer(); + Map<Vector2D<Integer, Integer>, Player> changes = new HashMap<>(); + changes.put(pos, currentPlayer); + board.add(currentPlayer, changes); + + playerManager.nextPlayer(); + return true; + } + + @Override + public boolean isMoveValid(Player player, Vector2D<Integer, Integer> pos) { + boolean positionOnBoard = (pos.getX() >= 0 && pos.getX() < board.getWidth()) && (pos.getY() >= 0 && pos.getY() < board.getHeight()); + boolean positionEmpty = (board.get(pos) == null); + return positionOnBoard && positionEmpty; + } + + @Override + public boolean isDraw() { + return !Objects.equals(board, null); + } + + @Override + public boolean isGameOver() { + return getWinners().size() < 1 || isDraw(); + } + + + private boolean isPlayer(Player player, Vector2D<Integer, Integer> pos) { + return Objects.equals(player, board.get(pos)); + } + @Override + public boolean isWinner(Player player) { + boolean topRow = isPlayer(player, new Vector2D<>(0,0)) && isPlayer(player, new Vector2D<>(0,1)) && isPlayer(player, new Vector2D<>(0,2)); + boolean middleRow = isPlayer(player, new Vector2D<>(1,0)) && isPlayer(player, new Vector2D<>(1,1)) && isPlayer(player, new Vector2D<>(1,2)); + boolean bottomRow = isPlayer(player, new Vector2D<>(2,0)) && isPlayer(player, new Vector2D<>(2,1)) && isPlayer(player, new Vector2D<>(2,2)); + + boolean leftColumn = isPlayer(player, new Vector2D<>(0,0)) && isPlayer(player, new Vector2D<>(0,1)) && isPlayer(player, new Vector2D<>(0,2)); + boolean midColumn = isPlayer(player, new Vector2D<>(1,0)) && isPlayer(player, new Vector2D<>(1,1)) && isPlayer(player, new Vector2D<>(1,2)); + boolean rightColumn = isPlayer(player, new Vector2D<>(2,0)) && isPlayer(player, new Vector2D<>(2,1)) && isPlayer(player, new Vector2D<>(2,2)); + + boolean lrCross = isPlayer(player, new Vector2D<>(0,0)) && isPlayer(player, new Vector2D<>(1,1)) && isPlayer(player, new Vector2D<>(2,2)); + boolean rlCross = isPlayer(player, new Vector2D<>(0,2)) && isPlayer(player, new Vector2D<>(1,1)) && isPlayer(player, new Vector2D<>(2,0)); + + return topRow || middleRow || bottomRow || leftColumn || midColumn || rightColumn || lrCross || rlCross; + } + + @Override + public List<Vector2D<Integer, Integer>> getValidMoves(Player player) { + List<Vector2D<Integer, Integer>> valid = new ArrayList<>(); + for (int y = 0; y < board.getHeight(); y++) { + for (int x = 0; x < board.getWidth(); x++) { + if (isMoveValid(player, new Vector2D<>(x, y))) valid.add(new Vector2D<>(x,y)); + } + } + return valid; + } + + public List<Player> getWinners(){ + List<Player> players = new ArrayList<>(); + for (Player player: playerManager.getPlayers()) { + if (isWinner(player)) { + players.add(player); + } + } + return players; + } + + @Override + public int getPlayerScore(Player player){ + if (isGameOver() && isWinner(player)) { + return 1; + } + return 0; + } +} +\ No newline at end of file diff --git a/src/main/java/nl/isygameclient/models/settings/GameCard.java b/src/main/java/nl/isygameclient/models/settings/GameCard.java @@ -0,0 +1,38 @@ +package nl.isygameclient.models.settings; + +public class GameCard { + + private String name; + private String viewSrc; + private String imgSrc; + + public GameCard(String name, String viewSrc, String imgSrc) { + this.name = name; + this.viewSrc = viewSrc; + this.imgSrc = imgSrc; + } + + public String getName() { + return name; + } + + public String getViewSrc() { + return viewSrc; + } + + public String getImgSrc() { + return imgSrc; + } + + public void setName(String name) { + this.name = name; + } + + public void setViewSrc(String viewSrc) { + this.viewSrc = viewSrc; + } + + public void setImgSrc(String imgSrc) { + this.imgSrc = imgSrc; + } +} diff --git a/src/main/java/nl/isygameclient/models/settings/Settings.java b/src/main/java/nl/isygameclient/models/settings/Settings.java @@ -0,0 +1,25 @@ +package nl.isygameclient.models.settings; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.Map; + + +/** + * This Class holds default settings for the application. + * It also provides the format of the settings that are saved to a file. + * + * @author A F Koens + */ +public class Settings { + public Map<String, StageSettings> stages = new HashMap<>(); + + public boolean tournamentMode = false; + + public ArrayList<GameCard> gameCards = new ArrayList<>(Arrays.asList( + new GameCard("TicTacToe", "views/games/tictactoe/TicTacToeMainMenu.fxml", "images/tictactoe_logo.png"), + new GameCard("Othello", "views/games/othello/OthelloSinglePlayer.fxml", "images/othello_logo.png"), + new GameCard("Starvation", null, null)) + ); +} diff --git a/src/main/java/nl/isygameclient/models/settings/StageSettings.java b/src/main/java/nl/isygameclient/models/settings/StageSettings.java @@ -0,0 +1,61 @@ +package nl.isygameclient.models.settings; + +import javafx.stage.Stage; + +public class StageSettings { + public String Title = "ISYGameClient"; + public double x = 0; + public double y = 0; + public double width = 680; + public double height = 480; + public double minWidth = 0; + public double minHeight = 0; + public double maxWidth = 1.7976931348623157E308; + public double maxHeight = 1.7976931348623157E308; + public boolean isIconified = false; + public boolean isAlwaysOnTop = false; + public boolean isFullScreen = true; + public boolean isResizable = true; + public boolean isMaximized = false; + public boolean isShowing = false; + + + public static StageSettings createSettingsFromStage(Stage stage) { + var stageSettings = new StageSettings(); + stageSettings.Title = stage.getTitle(); + stageSettings.x = stage.getX(); + stageSettings.y = stage.getY(); + stageSettings.width = stage.getWidth(); + stageSettings.height = stage.getHeight(); + stageSettings.minWidth = stage.getMinWidth(); + stageSettings.minHeight = stage.getMinHeight(); + stageSettings.maxWidth = stage.getMaxWidth(); + stageSettings.maxHeight = stage.getMaxHeight(); + stageSettings.isIconified = stage.isIconified(); + stageSettings.isAlwaysOnTop = stage.isAlwaysOnTop(); + stageSettings.isFullScreen = stage.isFullScreen(); + stageSettings.isResizable = stage.isResizable(); + stageSettings.isMaximized = stage.isMaximized(); + stageSettings.isShowing = stage.isShowing(); + return stageSettings; + } + + public static Stage configureStageFromSettings(StageSettings stageSettings) { + Stage stage = new Stage(); + stage.setTitle(stageSettings.Title); + stage.setX(stageSettings.x); + stage.setY(stageSettings.y); + stage.setWidth(stageSettings.width); + stage.setHeight(stageSettings.height); + stage.setMinWidth(stageSettings.minWidth); + stage.setMinHeight(stageSettings.minHeight); + stage.setMaxWidth(stageSettings.maxWidth); + stage.setMaxHeight(stageSettings.maxHeight); + stage.setIconified(stageSettings.isIconified); + stage.setAlwaysOnTop(stageSettings.isAlwaysOnTop); + stage.setFullScreen(stageSettings.isFullScreen); + stage.setResizable(stageSettings.isResizable); + stage.setMaximized(stageSettings.isMaximized); + return stage; + } +} diff --git a/src/main/java/nl/isygameclient/util/SettingsFileHandler.java b/src/main/java/nl/isygameclient/util/SettingsFileHandler.java @@ -2,7 +2,7 @@ package nl.isygameclient.util; import com.google.gson.Gson; import com.google.gson.GsonBuilder; -import nl.isygameclient.models.Settings.Settings; +import nl.isygameclient.models.settings.Settings; import java.io.FileWriter; import java.io.IOException; diff --git a/src/main/java/nl/isygameclient/util/StageHandler.java b/src/main/java/nl/isygameclient/util/StageHandler.java @@ -3,16 +3,15 @@ package nl.isygameclient.util; import javafx.fxml.FXMLLoader; import javafx.scene.Scene; import javafx.stage.Stage; -import nl.isygameclient.Application; -import nl.isygameclient.models.Settings.Settings; -import nl.isygameclient.models.Settings.StageSettings; +import nl.isygameclient.models.settings.Settings; +import nl.isygameclient.models.settings.StageSettings; import java.io.IOException; import java.util.HashMap; import java.util.Map; -import static nl.isygameclient.models.Settings.StageSettings.configureStageFromSettings; -import static nl.isygameclient.models.Settings.StageSettings.createSettingsFromStage; +import static nl.isygameclient.models.settings.StageSettings.configureStageFromSettings; +import static nl.isygameclient.models.settings.StageSettings.createSettingsFromStage; /** * This class is a singleton that stores and manages all the active stages in the application. diff --git a/src/main/java/nl/isygameclient/util/Vector2D.java b/src/main/java/nl/isygameclient/util/Vector2D.java @@ -0,0 +1,36 @@ + +package nl.isygameclient.util; + +public class Vector2D<X, Y> { + private X x; + private Y y; + + public Vector2D(X x, Y y) { + this.x = x; + this.y = y; + } + + public void setX(X x) { + this.x = x; + } + + public void setY(Y y) { + this.y = y; + } + + public X getX() { + return x; + } + + public Y getY() { + return y; + } + + @Override + public String toString() { + return "Vector2D{" + + "x=" + x + + ", y=" + y + + '}'; + } +} diff --git a/src/main/resources/nl/isygameclient/css/old_tokens.css b/src/main/resources/nl/isygameclient/css/old_tokens.css @@ -1,122 +0,0 @@ -* { - -md-source: #4caf50; - -md-color-black: #000000; - -md-color-white: #ffffff; - - /* error */ - -md-error50: #ffebee; - -md-error100: #ffcdd2; - -md-error200: #ef9a9a; - -md-error300: #e57373; - -md-error400: #ef5350; - -md-error500: #f44336; - -md-error600: #e53935; - -md-error700: #d32f2f; - -md-error800: #c62828; - -md-error900: #b71c1c; - - /* primary */ - -md-primary50: #e8f5e9; - -md-primary100: #c8e6c9; - -md-primary200: #a5d6a7; - -md-primary300: #81c784; - -md-primary400: #66bb6a; - -md-primary500: #4caf50; - -md-primary600: #43a047; - -md-primary700: #388e3c; - -md-primary800: #2e7d32; - -md-primary900: #1b5e20; - - /* secondary */ - -md-secondary50: #fff3e0; - -md-secondary100: #ffe0b2; - -md-secondary200: #ffcc80; - -md-secondary300: #ffb74d; - -md-secondary400: #ffa726; - -md-secondary500: #ff9800; - -md-secondary600: #fb8c00; - -md-secondary700: #f57c00; - -md-secondary800: #ef6c00; - -md-secondary900: #e65100; - - /* tertiary */ - -md-tertiary50: #e0f2f1; - -md-tertiary100: #b2dfdb; - -md-tertiary200: #80cbc4; - -md-tertiary300: #4db6ac; - -md-tertiary400: #26a69a; - -md-tertiary500: #009688; - -md-tertiary600: #00897b; - -md-tertiary700: #00796b; - -md-tertiary800: #00695c; - -md-tertiary900: #004d40; - - /* neutral */ - -md-neutral50: #fafafa; - -md-neutral100: #f5f5f5; - -md-neutral200: #eeeeee; - -md-neutral300: #e0e0e0; - -md-neutral400: #bdbdbd; - -md-neutral500: #787579; - -md-neutral600: #757575; - -md-neutral700: #616161; - -md-neutral800: #424242; - -md-neutral900: #212121; - - /* neutral-variant */ - -md-neutral-variant50: #eceff1; - -md-neutral-variant100: #cfd8dc; - -md-neutral-variant200: #b0bec5; - -md-neutral-variant300: #90a4ae; - -md-neutral-variant400: #78909c; - -md-neutral-variant500: #607d8b; - -md-neutral-variant600: #546e7a; - -md-neutral-variant700: #455a64; - -md-neutral-variant800: #37474f; - -md-neutral-variant900: #263238; - - /* Primary */ - -md-color-inverse-primary: -md-primary200; - -md-color-on-primary-container: -md-primary100; - -md-color-on-primary: -md-primary500; - -md-color-primary-container: -md-primary400; - -md-color-primary: -md-primary300; - - /* Secondary */ - -md-color-on-secondary-container: -md-secondary200; - -md-color-on-secondary: -md-secondary500; - -md-color-secondary-container: -md-secondary400; - -md-color-secondary: -md-secondary300; - - /* Tertiary */ - -md-color-on-tertiary-container: -md-error200; - -md-color-on-tertiary: -md-error400; - -md-color-tertiary-container: -md-error500; - -md-color-tertiary: -md-error300; - - /* Error */ - -md-color-on-error-container: -md-error300; - -md-color-on-error: -md-error500; - -md-color-error-container: -md-error600; - -md-color-error: -md-error300; - - /* Surface */ - -md-color-inverse-on-surface: -md-neutral500; - -md-color-inverse-surface: -md-neutral100; - - -md-color-on-surface-variant: -md-neutral-variant300; - -md-color-on-surface: -md-neutral300; - -md-color-surface-variant: -md-neutral700; - -md-color-surface: -md-neutral800; - - /* Background */ - -md-color-shadow: -md-color-black; - -md-color-outline: -md-neutral600; - -md-color-on-background: -md-neutral300; - -md-color-background: -md-neutral800; - - /* Text Colors */ - -md-color-text-highlight: -md-color-white; - -md-color-text-sub: -md-neutral400; - -md-color-text: -md-neutral200; -} diff --git a/src/main/resources/nl/isygameclient/css/style.css b/src/main/resources/nl/isygameclient/css/style.css @@ -1,6 +1,3 @@ -@import url(./views/game-select-menu.css); -@import url(./views/tictactoe.css); - .transparent { -fx-background-color: transparent; -} -\ No newline at end of file +} diff --git a/src/main/resources/nl/isygameclient/css/views/game-select-menu.css b/src/main/resources/nl/isygameclient/css/views/game-select-menu.css @@ -1,3 +1,4 @@ +@import url(../theme.css) /* Menu Bar */ .menu-bar { -fx-effect: dropshadow(three-pass-box, black, 10, 0, 0, 0); diff --git a/src/main/resources/nl/isygameclient/css/views/othello.css b/src/main/resources/nl/isygameclient/css/views/othello.css @@ -0,0 +1,5 @@ +@import url(../theme.css) + .othello-button { + -fx-background-color: #388E3C; + -fx-text-fill: -md-sys-color-on-surface-variant + } diff --git a/src/main/resources/nl/isygameclient/css/views/tictactoe.css b/src/main/resources/nl/isygameclient/css/views/tictactoe.css @@ -1,3 +1,4 @@ +@import url(../theme.css) .ttt-grid { -fx-border-color: -md-sys-color-surface-variant; -fx-border-width: 5; @@ -7,4 +8,4 @@ .ttt-button { -fx-background-color: -md-sys-color-surface-variant; -fx-text-fill: -md-sys-color-on-surface-variant - } -\ No newline at end of file + } diff --git a/src/main/resources/nl/isygameclient/views/GameSelectorMenu/GameCard.fxml b/src/main/resources/nl/isygameclient/views/GameSelectorMenu/GameCard.fxml @@ -1,23 +0,0 @@ -<?xml version="1.0" encoding="UTF-8"?> - -<?import java.lang.*?> -<?import javafx.scene.control.*?> -<?import javafx.scene.image.*?> -<?import javafx.scene.layout.*?> - -<VBox prefHeight="350.0" prefWidth="250.0" xmlns="http://javafx.com/javafx/16" xmlns:fx="http://javafx.com/fxml/1" - alignment="CENTER" fx:controller="nl.isygameclient.controllers.GameSeletorMenu.GameCardController" onMouseClicked="#onCardMouseClick"> - <styleClass> - <String fx:value="surface"/> - </styleClass> - <AnchorPane> - <ImageView fx:id="gameImage" fitWidth="250.0" preserveRatio="true"/> - </AnchorPane> - <Separator/> - <Label fx:id="gameTitle" alignment="TOP_RIGHT" text="Title"> - <styleClass> - <String fx:value="title-medium"/> - <String fx:value="on-surface-text"/> - </styleClass> - </Label> -</VBox> diff --git a/src/main/resources/nl/isygameclient/views/GameSelectorMenu/GameSelectorMenu.fxml b/src/main/resources/nl/isygameclient/views/GameSelectorMenu/GameSelectorMenu.fxml @@ -1,153 +0,0 @@ -<?xml version="1.0" encoding="UTF-8"?> - - -<?import com.jfoenix.controls.*?> -<?import de.jensd.fx.glyphs.materialicons.MaterialIconView?> -<?import javafx.geometry.*?> -<?import javafx.scene.control.Label?> -<?import javafx.scene.control.ListView?> -<?import javafx.scene.control.ScrollPane?> -<?import javafx.scene.control.TextField?> -<?import javafx.scene.layout.*?> -<?import java.lang.*?> -<AnchorPane minHeight="480.0" minWidth="600.0" prefHeight="1080.0" prefWidth="1920.0" styleClass="white" - stylesheets="@../../css/theme.css" xmlns="http://javafx.com/javafx/16" xmlns:fx="http://javafx.com/fxml/1" - fx:controller="nl.isygameclient.controllers.GameSeletorMenu.GameSelectorMenuController"> - <BorderPane AnchorPane.bottomAnchor="0.0" AnchorPane.leftAnchor="0.0" AnchorPane.rightAnchor="0.0" - AnchorPane.topAnchor="0.0"> - <top> - <HBox prefHeight="50.0" prefWidth="Infinity"> - <styleClass> - <String fx:value="menu-bar"/> - <String fx:value="background"/> - </styleClass> - - <!-- Menu Buttons --> - <JFXButton text="Store" disable="true" prefHeight="Infinity"> - <styleClass> - <String fx:value="title-medium"/> - <String fx:value="on-surface-text"/> - </styleClass> - </JFXButton> - - <JFXButton text="Games" prefHeight="Infinity"> - <styleClass> - <String fx:value="title-medium"/> - <String fx:value="on-surface-text"/> - </styleClass> - </JFXButton> - - <JFXButton text="User" disable="true" prefHeight="Infinity"> - <styleClass> - <String fx:value="title-medium"/> - <String fx:value="on-surface-text"/> - </styleClass> - </JFXButton> - - <Pane HBox.hgrow="ALWAYS"/> - <JFXButton textAlignment="RIGHT" text="Settings" prefHeight="Infinity"> - <styleClass> - <String fx:value="title-medium"/> - <String fx:value="on-surface-text"/> - </styleClass> - </JFXButton> - </HBox> - </top> - <left> - <AnchorPane> - <HBox AnchorPane.topAnchor="0.0" AnchorPane.bottomAnchor="0.0"> - - <!-- Search Container --> - <VBox prefWidth="200.0" AnchorPane.topAnchor="0.0" AnchorPane.bottomAnchor="0.0" - AnchorPane.leftAnchor="0.0"> - <padding> - <Insets left="5" right="5"/> - </padding> - <styleClass> - <String fx:value="search-container"/> - <String fx:value="surface"/> - </styleClass> - - <!-- Search Field --> - <StackPane> - <styleClass> - <String fx:value="search-bar"/> - <String fx:value="inverse-on-surface"/> - </styleClass> - <VBox.margin> - <Insets bottom="5.0" top="10.0"/> - </VBox.margin> - - - <TextField fx:id="searchBox" promptText="Search"> - <padding> - <Insets left="35" right="35"/> - </padding> - <styleClass> - <String fx:value="search-field"/> - <String fx:value="transparent"/> - </styleClass> - </TextField> - - <JFXButton StackPane.alignment="CENTER_LEFT" disable="true"> - <graphic> - <MaterialIconView StackPane.alignment="CENTER_LEFT" glyphName="SEARCH" - glyphSize="20" styleClass="icon"> - <styleClass> - <String fx:value="on-surface-fill"/> - </styleClass> - </MaterialIconView> - </graphic> - </JFXButton> - - <JFXButton onAction="#onClearSearchButtonClick" StackPane.alignment="CENTER_RIGHT"> - <graphic> - <MaterialIconView glyphName="CLOSE" glyphSize="16" styleClass="icon"> - <styleClass> - <String fx:value="on-surface-fill"/> - </styleClass> - </MaterialIconView> - </graphic> - </JFXButton> - </StackPane> - - <!-- Search List --> - <ListView fx:id="gamesList" VBox.vgrow="ALWAYS" styleClass="surface"> - <padding> - <Insets topRightBottomLeft="1"/> - </padding> - </ListView> - </VBox> - - - <!-- Game Detail Container --> - <VBox fx:id="gameDetail" prefWidth="400" HBox.hgrow="ALWAYS" alignment="TOP_CENTER" - styleClass="inverse-on-surface"> - <padding> - <Insets topRightBottomLeft="20"/> - </padding> - <Label text="Game Title"> - <styleClass> - <String fx:value="display-small"/> - <String fx:value="on-surface-text"/> - <String fx:value="on-surface-fill"/> - </styleClass> - </Label> - </VBox> - </HBox> - </AnchorPane> - </left> - <center> - <ScrollPane fitToWidth="true" fitToHeight="true" styleClass="surface-variant"> - <FlowPane fx:id="gameContainer" hgap="5" vgap="10" > - <styleClass> - <String fx:value="surface-variant"/> - </styleClass> - <padding> - <Insets topRightBottomLeft="20"/> - </padding> - </FlowPane> - </ScrollPane> - </center> - </BorderPane> -</AnchorPane> diff --git a/src/main/resources/nl/isygameclient/views/Games/TicTacToe/TicTacToeMainMenu.fxml b/src/main/resources/nl/isygameclient/views/Games/TicTacToe/TicTacToeMainMenu.fxml @@ -1,50 +0,0 @@ -<?xml version="1.0" encoding="UTF-8"?> - - -<?import com.jfoenix.controls.JFXButton?> -<?import javafx.geometry.Insets?> -<?import javafx.scene.control.Label?> -<?import javafx.scene.layout.AnchorPane?> -<?import javafx.scene.layout.VBox?> -<?import java.lang.*?> -<AnchorPane stylesheets="@../../../css/theme.css" xmlns="http://javafx.com/javafx" xmlns:fx="http://javafx.com/fxml" - fx:controller="nl.isygameclient.controllers.Games.TicTacToe.TicTacToeMainMenuController" - prefWidth="680" prefHeight="480"> - <VBox alignment="CENTER" AnchorPane.rightAnchor="0" AnchorPane.leftAnchor="0" AnchorPane.bottomAnchor="0" AnchorPane.topAnchor="0"> - <styleClass> - <String fx:value="surface"/> - </styleClass> - <Label text="Tic Tac Toe"> - <styleClass> - <String fx:value="display-large"/> - <String fx:value="on-surface-text"/> - </styleClass> - <padding> - <Insets topRightBottomLeft="50"/> - </padding> - </Label> - <VBox spacing="10" alignment="CENTER" maxWidth="200"> - <JFXButton text="Single Player" onAction="#onSinglePlayerButtonClick" prefWidth="Infinity"> - <styleClass> - <String fx:value="primary"/> - <String fx:value="on-primary-text"/> - <String fx:value="title-medium"/> - </styleClass> - </JFXButton> - <JFXButton text="Multiplayer" onAction="#onMultiplayerButtonClick" prefWidth="Infinity"> - <styleClass> - <String fx:value="primary"/> - <String fx:value="on-primary-text"/> - <String fx:value="title-medium"/> - </styleClass> - </JFXButton> - <JFXButton text="Exit" onAction="#onExitButtonClick" prefWidth="Infinity"> - <styleClass> - <String fx:value="primary"/> - <String fx:value="on-primary-text"/> - <String fx:value="title-medium"/> - </styleClass> - </JFXButton> - </VBox> - </VBox> -</AnchorPane> diff --git a/src/main/resources/nl/isygameclient/views/Games/TicTacToe/TicTacToeMultiPlayer.fxml b/src/main/resources/nl/isygameclient/views/Games/TicTacToe/TicTacToeMultiPlayer.fxml @@ -1,195 +0,0 @@ -<?xml version="1.0" encoding="UTF-8"?> - -<?import com.jfoenix.controls.*?> -<?import javafx.geometry.*?> -<?import javafx.scene.control.*?> -<?import javafx.scene.layout.*?> -<?import java.lang.*?> - -<BorderPane stylesheets="@../../../css/theme.css" xmlns="http://javafx.com/javafx/19" xmlns:fx="http://javafx.com/fxml/1" fx:controller="nl.isygameclient.controllers.Games.TicTacToe.TicTacToeMultiPlayerController"> - <styleClass> - <String fx:value="surface" /> - </styleClass> - <left> - <VBox prefWidth="200" spacing="20" styleClass="surface-variant"> - <padding> - <Insets topRightBottomLeft="20" /> - </padding> - - <!-- Playing As Controls --> - <VBox> - <Label text="Playing As"> - <styleClass> - <String fx:value="headline-small" /> - <String fx:value="on-surface-variant-text" /> - </styleClass> - </Label> - <Separator styleClass="on-surface-variant-text" /> - <JFXComboBox fx:id="playingAsCombo" onAction="#onPlayingAsComboSelect" prefWidth="Infinity"> - <styleClass> - <String fx:value="primary" /> - --> - <String fx:value="on-primary-text" /> - <String fx:value="body-large" /> - </styleClass> - </JFXComboBox> - </VBox> - <VBox> - <Label text="Opponent"> - <styleClass> - <String fx:value="headline-small" /> - <String fx:value="on-surface-variant-text" /> - </styleClass> - </Label> - <Separator styleClass="on-surface-variant-text" /> - <Label fx:id="playingAgainstLabel" prefWidth="Infinity"> - <styleClass> - <String fx:value="headline-small" /> - <String fx:value="on-surface-variant-text" /> - </styleClass> - </Label> - </VBox> - <VBox> - <children> - <Label text="Player"> - <styleClass> - <String fx:value="headline-small" /> - <String fx:value="on-surface-variant-text" /> - </styleClass> - </Label> - <Separator styleClass="on-surface-variant-text" /> - <TextField fx:id="nameField" promptText="Your Name" /> - <TextField fx:id="opponentField" promptText="Opponent (optional)" /> - </children> - </VBox> - - <!-- Best of Controls --> - <VBox> - <Label text="Server"> - <styleClass> - <String fx:value="headline-small" /> - <String fx:value="on-surface-variant-text" /> - </styleClass> - </Label> - <Separator styleClass="on-surface-variant-text" /> - <HBox prefHeight="100.0" prefWidth="200.0"> - <children> - <TextField fx:id="hostField" promptText="Host" /> - <TextField fx:id="portField" alignment="CENTER_RIGHT" promptText="Port" text="7789" /> - </children> - </HBox> - </VBox> - - <!-- Window Controls --> - <Pane VBox.vgrow="ALWAYS" /> - <VBox alignment="CENTER" spacing="10"> - <JFXButton onAction="#onNewGameButtonClick" prefWidth="Infinity" text="New Game"> - <styleClass> - <String fx:value="primary" /> - <String fx:value="on-primary-text" /> - <String fx:value="title-medium" /> - </styleClass> - </JFXButton> - <JFXButton onAction="#onMainMenuButtonClick" prefWidth="Infinity" text="Main Menu"> - <styleClass> - <String fx:value="primary" /> - <String fx:value="on-primary-text" /> - <String fx:value="title-medium" /> - </styleClass> - </JFXButton> - <JFXButton onAction="#onExitButtonClick" prefWidth="Infinity" text="Exit"> - <styleClass> - <String fx:value="primary" /> - <String fx:value="on-primary-text" /> - <String fx:value="title-medium" /> - </styleClass> - </JFXButton> - </VBox> - </VBox> - </left> - <!-- Game Field --> - <center> - <HBox alignment="CENTER"> - <padding> - <Insets topRightBottomLeft="20" /> - </padding> - - <!-- Left --> - <VBox alignment="TOP_CENTER" maxHeight="800" minWidth="150"> - <Label text="Score Player"> - <styleClass> - <String fx:value="headline-medium" /> - <String fx:value="on-surface-text" /> - </styleClass> - </Label> - <Label text="0"> - <styleClass> - <String fx:value="headline-medium" /> - <String fx:value="on-surface-text" /> - </styleClass> - </Label> - <Separator maxWidth="100" /> - </VBox> - - <!-- Center --> - <VBox alignment="CENTER" minWidth="640"> - <HBox alignment="CENTER"> - <padding> - <Insets topRightBottomLeft="5" /> - </padding> - <Label text="Current Player: "> - <styleClass> - <String fx:value="on-surface-text" /> - <String fx:value="headline-small" /> - </styleClass> - </Label> - <Label fx:id="currentPlayer"> - <styleClass> - <String fx:value="on-surface-text" /> - <String fx:value="headline-small" /> - </styleClass> - </Label> - </HBox> - <Separator maxWidth="100" /> - - <StackPane> - <VBox.margin> - <Insets left="20" right="20" top="20" /> - </VBox.margin> - <GridPane fx:id="grid" hgap="10" maxHeight="640" maxWidth="640" styleClass="ttt-grid" vgap="10"> - <padding> - <Insets topRightBottomLeft="10" /> - </padding> - </GridPane> - <Label fx:id="gameOverText" visible="false"> - <padding> - <Insets bottom="10" left="20" right="20" top="10" /> - </padding> - <styleClass> - <String fx:value="surface" /> - <String fx:value="display-large" /> - <String fx:value="on-surface-text" /> - </styleClass> - </Label> - </StackPane> - </VBox> - - <!-- Right --> - <VBox alignment="TOP_CENTER" maxHeight="800" minWidth="150"> - <Label text="Score Opponent"> - <styleClass> - <String fx:value="headline-medium" /> - <String fx:value="on-surface-text" /> - </styleClass> - </Label> - <Label text="0"> - <styleClass> - <String fx:value="headline-medium" /> - <String fx:value="on-surface-text" /> - </styleClass> - </Label> - <Separator maxWidth="100" /> - </VBox> - </HBox> - </center> -</BorderPane> -\ No newline at end of file diff --git a/src/main/resources/nl/isygameclient/views/Games/TicTacToe/TicTacToeSinglePlayer.fxml b/src/main/resources/nl/isygameclient/views/Games/TicTacToe/TicTacToeSinglePlayer.fxml @@ -1,196 +0,0 @@ -<?xml version="1.0" encoding="UTF-8"?> - -<?import com.jfoenix.controls.*?> -<?import javafx.geometry.*?> -<?import javafx.scene.control.*?> -<?import javafx.scene.layout.*?> -<?import java.lang.*?> -<BorderPane stylesheets="@../../../css/theme.css" xmlns="http://javafx.com/javafx/16" xmlns:fx="http://javafx.com/fxml/1" - fx:controller="nl.isygameclient.controllers.Games.TicTacToe.TicTacToeSinglePlayerController"> - <styleClass> - <String fx:value="surface"/> - </styleClass> - <left> - <VBox prefWidth="200" styleClass="surface-variant" spacing="20" > - <padding> - <Insets topRightBottomLeft="20"/> - </padding> - - <!-- Difficulty Controls --> - <VBox> - <Label text="Difficulty"> - <styleClass> - <String fx:value="headline-small"/> - <String fx:value="on-surface-variant-text"/> - </styleClass> - </Label> - <Separator styleClass="on-surface-variant-text"/> - <JFXComboBox fx:id="difficultyCombo" prefWidth="Infinity" onAction="#onDifficultyComboSelect"> - <styleClass> - <String fx:value="primary"/> - <String fx:value="on-primary-text"/> - <String fx:value="body-large"/> - </styleClass> - </JFXComboBox> - </VBox> - - <!-- Playing As Controls --> - <VBox> - <Label text="Playing As"> - <styleClass> - <String fx:value="headline-small"/> - <String fx:value="on-surface-variant-text"/> - </styleClass> - </Label> - <Separator styleClass="on-surface-variant-text"/> - <JFXComboBox fx:id="playingAsCombo" prefWidth="Infinity" onAction="#onPlayingAsComboSelect"> - <styleClass> - <String fx:value="primary"/>--> - <String fx:value="on-primary-text"/> - <String fx:value="body-large"/> - </styleClass> - </JFXComboBox> - </VBox> - - <!-- Opponents Controls --> - <VBox> - <Label text="Opponent"> - <styleClass> - <String fx:value="headline-small"/> - <String fx:value="on-surface-variant-text"/> - </styleClass> - </Label> - <Separator styleClass="on-surface-variant-text"/> - <JFXComboBox fx:id="opponentCombo" prefWidth="Infinity" onAction="#onOpponentComboSelect"> - <styleClass> - <String fx:value="primary"/>--> - <String fx:value="on-primary-text"/> - <String fx:value="body-large"/> - </styleClass> - </JFXComboBox> - </VBox> - - <!-- Best of Controls --> - <VBox> - <Label text="Best Of"> - <styleClass> - <String fx:value="headline-small"/> - <String fx:value="on-surface-variant-text"/> - </styleClass> - </Label> - <Separator styleClass="on-surface-variant-text"/> - </VBox> - - <!-- Window Controls --> - <Pane VBox.vgrow="ALWAYS"/> - <VBox spacing="10" alignment="CENTER"> - <JFXButton text="New Game" onAction="#onNewGameButtonClick" prefWidth="Infinity"> - <styleClass> - <String fx:value="primary"/> - <String fx:value="on-primary-text"/> - <String fx:value="title-medium"/> - </styleClass> - </JFXButton> - <JFXButton text="Main Menu" onAction="#onMainMenuButtonClick" prefWidth="Infinity"> - <styleClass> - <String fx:value="primary"/> - <String fx:value="on-primary-text"/> - <String fx:value="title-medium"/> - </styleClass> - </JFXButton> - <JFXButton text="Exit" onAction="#onExitButtonClick" prefWidth="Infinity"> - <styleClass> - <String fx:value="primary"/> - <String fx:value="on-primary-text"/> - <String fx:value="title-medium"/> - </styleClass> - </JFXButton> - </VBox> - </VBox> - </left> - <!-- Game Field --> - <center> - <HBox alignment="CENTER"> - <padding> - <Insets topRightBottomLeft="20"/> - </padding> - - <!-- Left --> - <VBox minWidth="150" maxHeight="800" alignment="TOP_CENTER"> - <Label text="Score Player"> - <styleClass> - <String fx:value="headline-medium"/> - <String fx:value="on-surface-text"/> - </styleClass> - </Label> - <Label text="0"> - <styleClass> - <String fx:value="headline-medium"/> - <String fx:value="on-surface-text"/> - </styleClass> - </Label> - <Separator maxWidth="100"/> - </VBox> - - <!-- Center --> - <VBox alignment="CENTER" minWidth="640"> - <HBox alignment="CENTER"> - <padding> - <Insets topRightBottomLeft="5"/> - </padding> - <Label text="Current Player: "> - <styleClass> - <String fx:value="on-surface-text"/> - <String fx:value="headline-small"/> - </styleClass> - </Label> - <Label fx:id="currentPlayer"> - <styleClass> - <String fx:value="on-surface-text"/> - <String fx:value="headline-small"/> - </styleClass> - </Label> - </HBox> - <Separator maxWidth="100"/> - - <StackPane> - <VBox.margin> - <Insets left="20" right="20" top="20"/> - </VBox.margin> - <GridPane fx:id="grid" styleClass="ttt-grid" vgap="10" hgap="10" maxWidth="640" maxHeight="640"> - <padding> - <Insets topRightBottomLeft="10"/> - </padding> - </GridPane> - <Label fx:id="gameOverText" visible="false"> - <padding> - <Insets top="10" bottom="10" left="20" right="20"/> - </padding> - <styleClass> - <String fx:value="surface"/> - <String fx:value="display-large"/> - <String fx:value="on-surface-text"/> - </styleClass> - </Label> - </StackPane> - </VBox> - - <!-- Right --> - <VBox minWidth="150" maxHeight="800" alignment="TOP_CENTER"> - <Label text="Score Opponent"> - <styleClass> - <String fx:value="headline-medium"/> - <String fx:value="on-surface-text"/> - </styleClass> - </Label> - <Label text="0"> - <styleClass> - <String fx:value="headline-medium"/> - <String fx:value="on-surface-text"/> - </styleClass> - </Label> - <Separator maxWidth="100"/> - </VBox> - </HBox> - </center> -</BorderPane> diff --git a/src/main/resources/nl/isygameclient/views/game_selector/GameCard.fxml b/src/main/resources/nl/isygameclient/views/game_selector/GameCard.fxml @@ -0,0 +1,23 @@ +<?xml version="1.0" encoding="UTF-8"?> + +<?import java.lang.*?> +<?import javafx.scene.control.*?> +<?import javafx.scene.image.*?> +<?import javafx.scene.layout.*?> + +<VBox prefHeight="350.0" prefWidth="250.0" xmlns="http://javafx.com/javafx/16" xmlns:fx="http://javafx.com/fxml/1" + alignment="CENTER" fx:controller="nl.isygameclient.controllers.game_selector.GameCardController" onMouseClicked="#onCardMouseClick"> + <styleClass> + <String fx:value="surface"/> + </styleClass> + <AnchorPane> + <ImageView fx:id="gameImage" fitWidth="250.0" preserveRatio="true"/> + </AnchorPane> + <Separator/> + <Label fx:id="gameTitle" alignment="TOP_RIGHT" text="Title"> + <styleClass> + <String fx:value="title-medium"/> + <String fx:value="on-surface-text"/> + </styleClass> + </Label> +</VBox> diff --git a/src/main/resources/nl/isygameclient/views/game_selector/GameSelector.fxml b/src/main/resources/nl/isygameclient/views/game_selector/GameSelector.fxml @@ -0,0 +1,153 @@ +<?xml version="1.0" encoding="UTF-8"?> + +<?import com.jfoenix.controls.*?> +<?import de.jensd.fx.glyphs.materialicons.MaterialIconView?> +<?import javafx.geometry.*?> +<?import javafx.scene.control.Label?> +<?import javafx.scene.control.ListView?> +<?import javafx.scene.control.ScrollPane?> +<?import javafx.scene.control.TextField?> +<?import javafx.scene.layout.*?> +<?import java.lang.*?> + +<AnchorPane minHeight="480.0" minWidth="600.0" prefHeight="1080.0" prefWidth="1920.0" styleClass="white" + stylesheets="@../../css/theme.css" xmlns="http://javafx.com/javafx/16" xmlns:fx="http://javafx.com/fxml/1" + fx:controller="nl.isygameclient.controllers.game_selector.GameSelectorController"> + <BorderPane AnchorPane.bottomAnchor="0.0" AnchorPane.leftAnchor="0.0" AnchorPane.rightAnchor="0.0" + AnchorPane.topAnchor="0.0"> + <top> + <HBox prefHeight="50.0" prefWidth="Infinity"> + <styleClass> + <String fx:value="menu-bar"/> + <String fx:value="background"/> + </styleClass> + + <!-- Menu Buttons --> + <JFXButton text="Store" disable="true" prefHeight="Infinity"> + <styleClass> + <String fx:value="title-medium"/> + <String fx:value="on-surface-text"/> + </styleClass> + </JFXButton> + + <JFXButton text="Games" prefHeight="Infinity"> + <styleClass> + <String fx:value="title-medium"/> + <String fx:value="on-surface-text"/> + </styleClass> + </JFXButton> + + <JFXButton text="User" disable="true" prefHeight="Infinity"> + <styleClass> + <String fx:value="title-medium"/> + <String fx:value="on-surface-text"/> + </styleClass> + </JFXButton> + + <Pane HBox.hgrow="ALWAYS"/> + <JFXButton textAlignment="RIGHT" text="Settings" prefHeight="Infinity"> + <styleClass> + <String fx:value="title-medium"/> + <String fx:value="on-surface-text"/> + </styleClass> + </JFXButton> + </HBox> + </top> + <left> + <AnchorPane> + <HBox AnchorPane.topAnchor="0.0" AnchorPane.bottomAnchor="0.0"> + + <!-- Search Container --> + <VBox prefWidth="200.0" AnchorPane.topAnchor="0.0" AnchorPane.bottomAnchor="0.0" + AnchorPane.leftAnchor="0.0"> + <padding> + <Insets left="5" right="5"/> + </padding> + <styleClass> + <String fx:value="search-container"/> + <String fx:value="surface"/> + </styleClass> + + <!-- Search Field --> + <StackPane> + <styleClass> + <String fx:value="search-bar"/> + <String fx:value="inverse-on-surface"/> + </styleClass> + <VBox.margin> + <Insets bottom="5.0" top="10.0"/> + </VBox.margin> + + + <TextField fx:id="searchBox" promptText="Search"> + <padding> + <Insets left="35" right="35"/> + </padding> + <styleClass> + <String fx:value="search-field"/> + <String fx:value="transparent"/> + </styleClass> + </TextField> + + <JFXButton StackPane.alignment="CENTER_LEFT" disable="true"> + <graphic> + <MaterialIconView StackPane.alignment="CENTER_LEFT" glyphName="SEARCH" + glyphSize="20" styleClass="icon"> + <styleClass> + <String fx:value="on-surface-fill"/> + </styleClass> + </MaterialIconView> + </graphic> + </JFXButton> + + <JFXButton onAction="#onClearSearchButtonClick" StackPane.alignment="CENTER_RIGHT"> + <graphic> + <MaterialIconView glyphName="CLOSE" glyphSize="16" styleClass="icon"> + <styleClass> + <String fx:value="on-surface-fill"/> + </styleClass> + </MaterialIconView> + </graphic> + </JFXButton> + </StackPane> + + <!-- Search List --> + <ListView fx:id="gamesList" VBox.vgrow="ALWAYS" styleClass="surface"> + <padding> + <Insets topRightBottomLeft="1"/> + </padding> + </ListView> + </VBox> + + + <!-- Game Detail Container --> + <VBox fx:id="gameDetail" prefWidth="400" HBox.hgrow="ALWAYS" alignment="TOP_CENTER" + styleClass="inverse-on-surface"> + <padding> + <Insets topRightBottomLeft="20"/> + </padding> + <Label text="Game Title"> + <styleClass> + <String fx:value="display-small"/> + <String fx:value="on-surface-text"/> + <String fx:value="on-surface-fill"/> + </styleClass> + </Label> + </VBox> + </HBox> + </AnchorPane> + </left> + <center> + <ScrollPane fitToWidth="true" fitToHeight="true" styleClass="surface-variant"> + <FlowPane fx:id="gameContainer" hgap="5" vgap="10" > + <styleClass> + <String fx:value="surface-variant"/> + </styleClass> + <padding> + <Insets topRightBottomLeft="20"/> + </padding> + </FlowPane> + </ScrollPane> + </center> + </BorderPane> +</AnchorPane> diff --git a/src/main/resources/nl/isygameclient/views/games/othello/OthelloSinglePlayer.fxml b/src/main/resources/nl/isygameclient/views/games/othello/OthelloSinglePlayer.fxml @@ -0,0 +1,55 @@ +<?xml version="1.0" encoding="UTF-8"?> + +<?import java.lang.*?> +<?import javafx.scene.layout.*?> + +<?import javafx.geometry.Insets?> +<?import com.jfoenix.controls.JFXButton?> +<BorderPane xmlns="http://javafx.com/javafx" + xmlns:fx="http://javafx.com/fxml" + stylesheets="@../../../css/views/othello.css" + fx:controller="nl.isygameclient.controllers.games.othello.OthelloSinglePlayerController" + prefHeight="400.0" prefWidth="600.0"> + <styleClass> + <String fx:value="surface"/> + </styleClass> + <left> + <VBox prefWidth="200" styleClass="surface-variant" spacing="20"> + <padding> + <Insets topRightBottomLeft="20"/> + </padding> + <!-- Window Controls --> + <Pane VBox.vgrow="ALWAYS"/> + <VBox spacing="10" alignment="CENTER"> + <JFXButton text="New Game" onAction="#onNewGameButtonClick" prefWidth="Infinity"> + <styleClass> + <String fx:value="primary"/> + <String fx:value="on-primary-text"/> + <String fx:value="title-medium"/> + </styleClass> + </JFXButton> + <JFXButton text="Main Menu" onAction="#onMainMenuButtonClick" prefWidth="Infinity"> + <styleClass> + <String fx:value="primary"/> + <String fx:value="on-primary-text"/> + <String fx:value="title-medium"/> + </styleClass> + </JFXButton> + <JFXButton text="Exit" onAction="#onExitButtonClick" prefWidth="Infinity"> + <styleClass> + <String fx:value="primary"/> + <String fx:value="on-primary-text"/> + <String fx:value="title-medium"/> + </styleClass> + </JFXButton> + </VBox> + </VBox> + </left> + <center> + <GridPane fx:id="boardGrid" alignment="CENTER" vgap="2" hgap="2"> + <padding> + <Insets topRightBottomLeft="10"/> + </padding> + </GridPane> + </center> +</BorderPane> diff --git a/src/main/resources/nl/isygameclient/views/games/tictactoe/TicTacToeMainMenu.fxml b/src/main/resources/nl/isygameclient/views/games/tictactoe/TicTacToeMainMenu.fxml @@ -0,0 +1,50 @@ +<?xml version="1.0" encoding="UTF-8"?> + + +<?import com.jfoenix.controls.JFXButton?> +<?import javafx.geometry.Insets?> +<?import javafx.scene.control.Label?> +<?import javafx.scene.layout.AnchorPane?> +<?import javafx.scene.layout.VBox?> +<?import java.lang.*?> +<AnchorPane stylesheets="@../../../css/theme.css" xmlns="http://javafx.com/javafx" xmlns:fx="http://javafx.com/fxml" + fx:controller="nl.isygameclient.controllers.games.tictactoe.TicTacToeMainMenuController" + prefWidth="680" prefHeight="480"> + <VBox alignment="CENTER" AnchorPane.rightAnchor="0" AnchorPane.leftAnchor="0" AnchorPane.bottomAnchor="0" AnchorPane.topAnchor="0"> + <styleClass> + <String fx:value="surface"/> + </styleClass> + <Label text="Tic Tac Toe"> + <styleClass> + <String fx:value="display-large"/> + <String fx:value="on-surface-text"/> + </styleClass> + <padding> + <Insets topRightBottomLeft="50"/> + </padding> + </Label> + <VBox spacing="10" alignment="CENTER" maxWidth="200"> + <JFXButton text="Single Player" onAction="#onSinglePlayerButtonClick" prefWidth="Infinity"> + <styleClass> + <String fx:value="primary"/> + <String fx:value="on-primary-text"/> + <String fx:value="title-medium"/> + </styleClass> + </JFXButton> + <JFXButton text="Multiplayer" onAction="#onMultiplayerButtonClick" prefWidth="Infinity"> + <styleClass> + <String fx:value="primary"/> + <String fx:value="on-primary-text"/> + <String fx:value="title-medium"/> + </styleClass> + </JFXButton> + <JFXButton text="Exit" onAction="#onExitButtonClick" prefWidth="Infinity"> + <styleClass> + <String fx:value="primary"/> + <String fx:value="on-primary-text"/> + <String fx:value="title-medium"/> + </styleClass> + </JFXButton> + </VBox> + </VBox> +</AnchorPane> diff --git a/src/main/resources/nl/isygameclient/views/games/tictactoe/TicTacToeMultiPlayer.fxml b/src/main/resources/nl/isygameclient/views/games/tictactoe/TicTacToeMultiPlayer.fxml @@ -0,0 +1,195 @@ +<?xml version="1.0" encoding="UTF-8"?> + +<?import com.jfoenix.controls.*?> +<?import javafx.geometry.*?> +<?import javafx.scene.control.*?> +<?import javafx.scene.layout.*?> +<?import java.lang.*?> + +<BorderPane stylesheets="@../../../css/theme.css" xmlns="http://javafx.com/javafx/19" xmlns:fx="http://javafx.com/fxml/1" fx:controller="nl.isygameclient.controllers.games.tictactoe.TicTacToeMultiPlayerController"> + <styleClass> + <String fx:value="surface" /> + </styleClass> + <left> + <VBox prefWidth="200" spacing="20" styleClass="surface-variant"> + <padding> + <Insets topRightBottomLeft="20" /> + </padding> + + <!-- Playing As Controls --> + <VBox> + <Label text="Playing As"> + <styleClass> + <String fx:value="headline-small" /> + <String fx:value="on-surface-variant-text" /> + </styleClass> + </Label> + <Separator styleClass="on-surface-variant-text" /> + <JFXComboBox fx:id="playingAsCombo" onAction="#onPlayingAsComboSelect" prefWidth="Infinity"> + <styleClass> + <String fx:value="primary" /> + --> + <String fx:value="on-primary-text" /> + <String fx:value="body-large" /> + </styleClass> + </JFXComboBox> + </VBox> + <VBox> + <Label text="Opponent"> + <styleClass> + <String fx:value="headline-small" /> + <String fx:value="on-surface-variant-text" /> + </styleClass> + </Label> + <Separator styleClass="on-surface-variant-text" /> + <Label fx:id="playingAgainstLabel" prefWidth="Infinity"> + <styleClass> + <String fx:value="headline-small" /> + <String fx:value="on-surface-variant-text" /> + </styleClass> + </Label> + </VBox> + <VBox> + <children> + <Label text="Player"> + <styleClass> + <String fx:value="headline-small" /> + <String fx:value="on-surface-variant-text" /> + </styleClass> + </Label> + <Separator styleClass="on-surface-variant-text" /> + <TextField fx:id="nameField" promptText="Your Name" /> + <TextField fx:id="opponentField" promptText="Opponent (optional)" /> + </children> + </VBox> + + <!-- Best of Controls --> + <VBox> + <Label text="Server"> + <styleClass> + <String fx:value="headline-small" /> + <String fx:value="on-surface-variant-text" /> + </styleClass> + </Label> + <Separator styleClass="on-surface-variant-text" /> + <HBox prefHeight="100.0" prefWidth="200.0"> + <children> + <TextField fx:id="hostField" promptText="Host" /> + <TextField fx:id="portField" alignment="CENTER_RIGHT" promptText="Port" text="7789" /> + </children> + </HBox> + </VBox> + + <!-- Window Controls --> + <Pane VBox.vgrow="ALWAYS" /> + <VBox alignment="CENTER" spacing="10"> + <JFXButton onAction="#onNewGameButtonClick" prefWidth="Infinity" text="New Game"> + <styleClass> + <String fx:value="primary" /> + <String fx:value="on-primary-text" /> + <String fx:value="title-medium" /> + </styleClass> + </JFXButton> + <JFXButton onAction="#onMainMenuButtonClick" prefWidth="Infinity" text="Main Menu"> + <styleClass> + <String fx:value="primary" /> + <String fx:value="on-primary-text" /> + <String fx:value="title-medium" /> + </styleClass> + </JFXButton> + <JFXButton onAction="#onExitButtonClick" prefWidth="Infinity" text="Exit"> + <styleClass> + <String fx:value="primary" /> + <String fx:value="on-primary-text" /> + <String fx:value="title-medium" /> + </styleClass> + </JFXButton> + </VBox> + </VBox> + </left> + <!-- Game Field --> + <center> + <HBox alignment="CENTER"> + <padding> + <Insets topRightBottomLeft="20" /> + </padding> + + <!-- Left --> + <VBox alignment="TOP_CENTER" maxHeight="800" minWidth="150"> + <Label text="Score Player"> + <styleClass> + <String fx:value="headline-medium" /> + <String fx:value="on-surface-text" /> + </styleClass> + </Label> + <Label text="0"> + <styleClass> + <String fx:value="headline-medium" /> + <String fx:value="on-surface-text" /> + </styleClass> + </Label> + <Separator maxWidth="100" /> + </VBox> + + <!-- Center --> + <VBox alignment="CENTER" minWidth="640"> + <HBox alignment="CENTER"> + <padding> + <Insets topRightBottomLeft="5" /> + </padding> + <Label text="Current Player: "> + <styleClass> + <String fx:value="on-surface-text" /> + <String fx:value="headline-small" /> + </styleClass> + </Label> + <Label fx:id="currentPlayer"> + <styleClass> + <String fx:value="on-surface-text" /> + <String fx:value="headline-small" /> + </styleClass> + </Label> + </HBox> + <Separator maxWidth="100" /> + + <StackPane> + <VBox.margin> + <Insets left="20" right="20" top="20" /> + </VBox.margin> + <GridPane fx:id="grid" hgap="10" maxHeight="640" maxWidth="640" styleClass="ttt-grid" vgap="10"> + <padding> + <Insets topRightBottomLeft="10" /> + </padding> + </GridPane> + <Label fx:id="gameOverText" visible="false"> + <padding> + <Insets bottom="10" left="20" right="20" top="10" /> + </padding> + <styleClass> + <String fx:value="surface" /> + <String fx:value="display-large" /> + <String fx:value="on-surface-text" /> + </styleClass> + </Label> + </StackPane> + </VBox> + + <!-- Right --> + <VBox alignment="TOP_CENTER" maxHeight="800" minWidth="150"> + <Label text="Score Opponent"> + <styleClass> + <String fx:value="headline-medium" /> + <String fx:value="on-surface-text" /> + </styleClass> + </Label> + <Label text="0"> + <styleClass> + <String fx:value="headline-medium" /> + <String fx:value="on-surface-text" /> + </styleClass> + </Label> + <Separator maxWidth="100" /> + </VBox> + </HBox> + </center> +</BorderPane> +\ No newline at end of file diff --git a/src/main/resources/nl/isygameclient/views/games/tictactoe/TicTacToeSinglePlayer.fxml b/src/main/resources/nl/isygameclient/views/games/tictactoe/TicTacToeSinglePlayer.fxml @@ -0,0 +1,196 @@ +<?xml version="1.0" encoding="UTF-8"?> + +<?import com.jfoenix.controls.*?> +<?import javafx.geometry.*?> +<?import javafx.scene.control.*?> +<?import javafx.scene.layout.*?> +<?import java.lang.*?> +<BorderPane stylesheets="@../../../css/theme.css" xmlns="http://javafx.com/javafx/16" xmlns:fx="http://javafx.com/fxml/1" + fx:controller="nl.isygameclient.controllers.games.tictactoe.TicTacToeSinglePlayerController"> + <styleClass> + <String fx:value="surface"/> + </styleClass> + <left> + <VBox prefWidth="200" styleClass="surface-variant" spacing="20" > + <padding> + <Insets topRightBottomLeft="20"/> + </padding> + +<!-- <!– Difficulty Controls –>--> +<!-- <VBox>--> +<!-- <Label text="Difficulty">--> +<!-- <styleClass>--> +<!-- <String fx:value="headline-small"/>--> +<!-- <String fx:value="on-surface-variant-text"/>--> +<!-- </styleClass>--> +<!-- </Label>--> +<!-- <Separator styleClass="on-surface-variant-text"/>--> +<!-- <JFXComboBox fx:id="difficultyCombo" prefWidth="Infinity" onAction="#onDifficultyComboSelect">--> +<!-- <styleClass>--> +<!-- <String fx:value="primary"/>--> +<!-- <String fx:value="on-primary-text"/>--> +<!-- <String fx:value="body-large"/>--> +<!-- </styleClass>--> +<!-- </JFXComboBox>--> +<!-- </VBox>--> + +<!-- <!– Playing As Controls –>--> +<!-- <VBox>--> +<!-- <Label text="Playing As">--> +<!-- <styleClass>--> +<!-- <String fx:value="headline-small"/>--> +<!-- <String fx:value="on-surface-variant-text"/>--> +<!-- </styleClass>--> +<!-- </Label>--> +<!-- <Separator styleClass="on-surface-variant-text"/>--> +<!-- <JFXComboBox fx:id="playingAsCombo" prefWidth="Infinity" onAction="#onPlayingAsComboSelect">--> +<!-- <styleClass>--> +<!-- <String fx:value="primary"/>–>--> +<!-- <String fx:value="on-primary-text"/>--> +<!-- <String fx:value="body-large"/>--> +<!-- </styleClass>--> +<!-- </JFXComboBox>--> +<!-- </VBox>--> + +<!-- <!– Opponents Controls –>--> +<!-- <VBox>--> +<!-- <Label text="Opponent">--> +<!-- <styleClass>--> +<!-- <String fx:value="headline-small"/>--> +<!-- <String fx:value="on-surface-variant-text"/>--> +<!-- </styleClass>--> +<!-- </Label>--> +<!-- <Separator styleClass="on-surface-variant-text"/>--> +<!-- <JFXComboBox fx:id="opponentCombo" prefWidth="Infinity" onAction="#onOpponentComboSelect">--> +<!-- <styleClass>--> +<!-- <String fx:value="primary"/>–>--> +<!-- <String fx:value="on-primary-text"/>--> +<!-- <String fx:value="body-large"/>--> +<!-- </styleClass>--> +<!-- </JFXComboBox>--> +<!-- </VBox>--> + +<!-- <!– Best of Controls –>--> +<!-- <VBox>--> +<!-- <Label text="Best Of">--> +<!-- <styleClass>--> +<!-- <String fx:value="headline-small"/>--> +<!-- <String fx:value="on-surface-variant-text"/>--> +<!-- </styleClass>--> +<!-- </Label>--> +<!-- <Separator styleClass="on-surface-variant-text"/>--> +<!-- </VBox>--> + + <!-- Window Controls --> + <Pane VBox.vgrow="ALWAYS"/> + <VBox spacing="10" alignment="CENTER"> + <JFXButton text="New Game" onAction="#onNewGameButtonClick" prefWidth="Infinity"> + <styleClass> + <String fx:value="primary"/> + <String fx:value="on-primary-text"/> + <String fx:value="title-medium"/> + </styleClass> + </JFXButton> + <JFXButton text="Main Menu" onAction="#onMainMenuButtonClick" prefWidth="Infinity"> + <styleClass> + <String fx:value="primary"/> + <String fx:value="on-primary-text"/> + <String fx:value="title-medium"/> + </styleClass> + </JFXButton> + <JFXButton text="Exit" onAction="#onExitButtonClick" prefWidth="Infinity"> + <styleClass> + <String fx:value="primary"/> + <String fx:value="on-primary-text"/> + <String fx:value="title-medium"/> + </styleClass> + </JFXButton> + </VBox> + </VBox> + </left> + <!-- Game Field --> + <center> + <HBox alignment="CENTER"> + <padding> + <Insets topRightBottomLeft="20"/> + </padding> + + <!-- Left --> + <VBox minWidth="150" maxHeight="800" alignment="TOP_CENTER"> + <Label text="Score Player"> + <styleClass> + <String fx:value="headline-medium"/> + <String fx:value="on-surface-text"/> + </styleClass> + </Label> + <Label text="0"> + <styleClass> + <String fx:value="headline-medium"/> + <String fx:value="on-surface-text"/> + </styleClass> + </Label> + <Separator maxWidth="100"/> + </VBox> + + <!-- Center --> + <VBox alignment="CENTER" minWidth="640"> + <HBox alignment="CENTER"> + <padding> + <Insets topRightBottomLeft="5"/> + </padding> + <Label text="Current Player: "> + <styleClass> + <String fx:value="on-surface-text"/> + <String fx:value="headline-small"/> + </styleClass> + </Label> + <Label fx:id="currentPlayer"> + <styleClass> + <String fx:value="on-surface-text"/> + <String fx:value="headline-small"/> + </styleClass> + </Label> + </HBox> + <Separator maxWidth="100"/> + + <StackPane> + <VBox.margin> + <Insets left="20" right="20" top="20"/> + </VBox.margin> + <GridPane fx:id="grid" styleClass="ttt-grid" vgap="10" hgap="10" maxWidth="640" maxHeight="640"> + <padding> + <Insets topRightBottomLeft="10"/> + </padding> + </GridPane> + <Label fx:id="gameOverText" visible="false"> + <padding> + <Insets top="10" bottom="10" left="20" right="20"/> + </padding> + <styleClass> + <String fx:value="surface"/> + <String fx:value="display-large"/> + <String fx:value="on-surface-text"/> + </styleClass> + </Label> + </StackPane> + </VBox> + + <!-- Right --> + <VBox minWidth="150" maxHeight="800" alignment="TOP_CENTER"> + <Label text="Score Opponent"> + <styleClass> + <String fx:value="headline-medium"/> + <String fx:value="on-surface-text"/> + </styleClass> + </Label> + <Label text="0"> + <styleClass> + <String fx:value="headline-medium"/> + <String fx:value="on-surface-text"/> + </styleClass> + </Label> + <Separator maxWidth="100"/> + </VBox> + </HBox> + </center> +</BorderPane>