Ludo-3D/Assets/Scripts/Gameplay/GameplayManager.cs

1432 lines
55 KiB
C#
Raw Permalink Normal View History

using TMPro;
using System;
2026-01-21 20:27:45 +05:30
using System.Linq;
using DG.Tweening;
2026-01-21 20:27:45 +05:30
using UnityEngine;
using System.Collections.Generic;
2026-02-12 21:34:08 +05:30
using System.Threading;
using UnityEngine.Serialization;
2026-01-21 20:27:45 +05:30
2026-01-27 20:55:55 +05:30
public enum BotMove
{
FinishingPathMove = 0,
SafeMove = 1,
AttackMove = 2,
NormalMove = 3,
NoMoves = 4
}
2026-01-21 20:27:45 +05:30
public class GameplayManager : MonoBehaviour, IBase, IBootLoader, IDataLoader
{
2026-02-04 12:32:18 +05:30
[SerializeField] private bool isDebugAITest = false;
[SerializeField] private bool isDebugPlayerTest = false;
2026-02-04 12:32:18 +05:30
public bool IsDebugAITest => isDebugAITest;
public bool IsDebugPlayerTest => isDebugPlayerTest;
[SerializeField] DiceRollHandler diceRollHandler;
2026-01-28 20:57:35 +05:30
[SerializeField] private int diceValue = 0;
2026-02-12 21:34:08 +05:30
[SerializeField] private float diceRollDelayForBot = 0.5f;
[SerializeField] private float diceRollDelayForUser = 0.5f;
[SerializeField] private int maxDiceSixRollCounter = 2;
[SerializeField] private int totalStepsForCharacter = 57;
[SerializeField] private int currentPlayerTurnMaxTime = 5;
2026-02-12 21:34:08 +05:30
[SerializeField] private int currentPlayerSelectionMaxTime = 5;
2026-01-28 20:57:35 +05:30
2026-01-26 19:48:55 +05:30
[SerializeField] private TextMeshProUGUI diceText;
2026-02-04 12:09:04 +05:30
#if UNITY_EDITOR
2026-01-21 20:27:45 +05:30
[SerializeField] private Transform pointerDebug;
[SerializeField] private MeshRenderer pointerMeshRend;
[SerializeField] private Material turnMat;
[SerializeField] private Material selectMat;
2026-02-04 12:09:04 +05:30
#endif
2026-01-21 20:27:45 +05:30
[SerializeField] private PlayerGameData[] playerGameDatas;
[SerializeField] private PlayerBaseHandler playerBaseHandler;
2026-01-21 20:27:45 +05:30
2026-02-02 19:27:17 +05:30
public PlayerBaseHandler PlayerBaseHandler => playerBaseHandler;
private PlayerType currentPlayerTypeTurn;
private TimerSystem currentPlayerTurnTimer;
private TimerSystem savedCurrentPlayerTurnTimerInst;
2026-01-21 20:27:45 +05:30
private int currentPlayerTurnIndex = 0;
2026-01-30 21:51:33 +05:30
2026-01-27 20:55:55 +05:30
private List<PlayerType> allPlayerTypes = new List<PlayerType>();
private List<PlayerType> botTypesInGame = new List<PlayerType>();
2026-01-21 20:27:45 +05:30
2026-01-30 21:51:33 +05:30
private List<PlayerData> playerDatas = new List<PlayerData>();
2026-01-21 20:27:45 +05:30
2026-01-30 21:51:33 +05:30
private Dictionary<PlayerType, PlayerGameData> playerGameDatasDict = new Dictionary<PlayerType, PlayerGameData>();
2026-01-30 19:04:14 +05:30
private Dictionary<PlayerType, Dictionary<int, BotMove>> botRuntimeMovementData = new Dictionary<PlayerType, Dictionary<int, BotMove>>();
2026-01-27 20:55:55 +05:30
2026-02-02 20:12:00 +05:30
public TilesManager TilesManager
{
get; private set;
}
2026-01-30 22:17:18 +05:30
private UIManager uIManager;
public GameManager GameManager
{
get; private set;
}
2026-02-02 17:21:38 +05:30
private GameModeHandler gameModeHandler;
private SoundManager soundManager;
2026-01-21 20:27:45 +05:30
private int diceRolledValue;
private bool CanRollDiceAgain = false; // used for when you get a 6 or when you reach the finish point
private int diceSixRollCounter = 0;
2026-01-21 20:27:45 +05:30
2026-02-10 19:24:08 +05:30
private List<PlayerPawn> availPlayersToMove = new List<PlayerPawn>();
private bool canSwitchPlayer = true;
public bool CanRollDiceForUser
{
get; private set;
}
2026-01-30 21:51:33 +05:30
public int TotalPlayersInGame
{
get; private set;
}
public List<PlayerData> PlayerDatas => playerDatas;
public List<PlayerType> PlayerTypesCollection => allPlayerTypes;
2026-01-30 19:04:14 +05:30
public Action onGameResumed = null;
2026-01-21 20:27:45 +05:30
public void Initialize()
{
InterfaceManager.Instance?.RegisterInterface<GameplayManager>(this);
DOTween.useSafeMode = false;
2026-01-21 20:27:45 +05:30
}
public void InitializeData()
{
2026-02-02 20:12:00 +05:30
TilesManager = InterfaceManager.Instance.GetInterfaceInstance<TilesManager>();
2026-01-30 22:17:18 +05:30
uIManager = InterfaceManager.Instance.GetInterfaceInstance<UIManager>();
GameManager = InterfaceManager.Instance.GetInterfaceInstance<GameManager>();
2026-02-02 17:21:38 +05:30
gameModeHandler = InterfaceManager.Instance.GetInterfaceInstance<GameModeHandler>();
soundManager = InterfaceManager.Instance.GetInterfaceInstance<SoundManager>();
2026-01-26 19:48:55 +05:30
}
2026-01-30 21:51:33 +05:30
public void InitPlayerTypesForPVP(List<PlayerType> types, List<string> names)
2026-01-26 19:48:55 +05:30
{
allPlayerTypes = new List<PlayerType>(types);
2026-02-02 17:21:38 +05:30
playerDatas = new List<PlayerData>();
2026-01-30 21:51:33 +05:30
TotalPlayersInGame = types.Count;
for (int i=0; i<types.Count; i++)
{
playerDatas.Add(new PlayerData
{
playerType = types[i],
playerName = names[i],
});
}
2026-02-02 20:12:00 +05:30
TilesManager.InitTilesData();
2026-01-30 22:06:51 +05:30
2026-01-27 20:55:55 +05:30
playerBaseHandler.InitPlayerTypes(allPlayerTypes);
InitCurrentGamePlayerInfo();
2026-01-21 20:27:45 +05:30
}
2026-01-30 21:51:33 +05:30
public void InitPlayerTypesForBotMatch(PlayerData selectedPlayerData, int botCount)
2026-01-26 19:48:55 +05:30
{
2026-01-30 22:06:51 +05:30
playerDatas = new List<PlayerData>();
botTypesInGame = new List<PlayerType>();
allPlayerTypes = new List<PlayerType>();
2026-01-27 20:55:55 +05:30
2026-01-30 21:51:33 +05:30
TotalPlayersInGame = botCount + 1;
AssignBotTypes(selectedPlayerData.playerType, botCount);
foreach (PlayerType playerType in Enum.GetValues(typeof(PlayerType)))
{
if (botTypesInGame.Contains(playerType))
{
allPlayerTypes.Add(playerType);
2026-01-31 00:05:18 +05:30
playerDatas.Add(new PlayerData { playerType = playerType, playerName = $"{playerType}" });
2026-01-30 21:51:33 +05:30
}
2026-01-30 22:29:00 +05:30
else if (selectedPlayerData.playerType == playerType)
2026-01-30 21:51:33 +05:30
{
allPlayerTypes.Add(selectedPlayerData.playerType);
playerDatas.Add(new PlayerData { playerType = selectedPlayerData.playerType, playerName = $"{selectedPlayerData.playerName}" });
}
}
2026-01-27 20:55:55 +05:30
2026-02-02 20:12:00 +05:30
TilesManager.InitTilesData();
2026-01-27 20:55:55 +05:30
playerBaseHandler.InitPlayerTypes(allPlayerTypes);
InitCurrentGamePlayerInfo();
2026-01-28 16:57:27 +05:30
InitBotRuntimeData();
2026-01-30 21:51:33 +05:30
AssignPlayerAndBotStates(selectedPlayerData.playerType);
2026-01-30 19:04:14 +05:30
// SetCanRollDiceForUser(!botTypesInGame.Contains(allPlayerTypes[0]));
2026-01-30 21:51:33 +05:30
if (botTypesInGame.Contains(allPlayerTypes[0]))
2026-02-04 12:32:18 +05:30
HandleDiceRollWithDelay();
}
private void HandleDiceRollWithDelay()
{
2026-02-12 21:34:08 +05:30
Invoke(nameof(RollDiceForBot), diceRollDelayForBot);
2026-01-27 20:55:55 +05:30
}
private void InitBotRuntimeData()
{
2026-01-30 22:06:51 +05:30
botRuntimeMovementData = new Dictionary<PlayerType, Dictionary<int, BotMove>>();
2026-01-27 20:55:55 +05:30
foreach (var botType in botTypesInGame)
{
PlayerGameData playerGameData = playerGameDatasDict[botType];
if (!botRuntimeMovementData.ContainsKey(botType))
{
botRuntimeMovementData.Add(botType, new Dictionary<int, BotMove>());
}
foreach (var pawn in playerGameData.playerPawnsDict)
{
2026-01-28 16:57:27 +05:30
botRuntimeMovementData[botType].Add(pawn.Value.PlayerId, BotMove.NoMoves);
2026-01-27 20:55:55 +05:30
}
}
}
private void AssignPlayerAndBotStates(PlayerType selectedPlayerType)
{
PlayerBase playerBase = playerBaseHandler.GetPlayerBase(selectedPlayerType);
playerBase.AssignBotState(PawnType.User);
foreach (PlayerType playerType in botTypesInGame)
{
playerBase = playerBaseHandler.GetPlayerBase(playerType);
playerBase.AssignBotState(PawnType.Bot);
}
}
private void AssignBotTypes(PlayerType selectedPlayerType, int botCount)
{
int addedBots = 0;
2026-01-26 19:48:55 +05:30
if (botCount == 1)
{
2026-01-27 20:55:55 +05:30
var playerType = (PlayerType)((int)selectedPlayerType + 2 < Enum.GetValues(typeof(PlayerType)).Length ?
2026-01-26 19:48:55 +05:30
(int)selectedPlayerType + 2 : (int)selectedPlayerType - 2);
2026-01-27 20:55:55 +05:30
botTypesInGame.Add(playerType);
}
else if (botCount == 2)
{
foreach (PlayerType enumType in Enum.GetValues(typeof(PlayerType)))
{
if (enumType == selectedPlayerType) continue;
botTypesInGame.Add(enumType);
addedBots++;
if (addedBots == botCount) break;
}
2026-01-26 19:48:55 +05:30
}
else
{
2026-01-27 20:55:55 +05:30
foreach (PlayerType enumType in Enum.GetValues(typeof(PlayerType)))
{
if (enumType == selectedPlayerType) continue;
botTypesInGame.Add(enumType);
}
2026-01-26 19:48:55 +05:30
}
}
public void UpdateCurrentPlayerTurn(PlayerType playerType)
{
playerBaseHandler.ShowSelectedPlayerBase(currentPlayerTypeTurn, false);
currentPlayerTypeTurn = playerType;
playerBaseHandler.ShowSelectedPlayerBase(currentPlayerTypeTurn, true);
2026-02-12 21:34:08 +05:30
}
2026-02-12 21:34:08 +05:30
private void UpdateDiceView()
{
diceRollHandler.DiceView.SetDiceButtonInteraction(true);
diceRollHandler.DiceView.ShowDefaultSprite();
UpdateResponseTimerForUser(currentPlayerMaxTime: currentPlayerTurnMaxTime, () => RollDiceForUser());
}
private void UpdateDiceViewForBot()
{
UpdateDiceView();
HandleDiceRollWithDelay();
}
2026-02-12 21:34:08 +05:30
private void UpdateResponseTimerForUser(int currentPlayerMaxTime, Action onComplete)
{
if (gameModeHandler.CurrentGameModeType == GameModeType.Bot)
{
bool isBotTurn = botTypesInGame.Contains(currentPlayerTypeTurn);
// SetCanRollDiceForUser(!isBotTurn); // TODO :: Need to change
2026-02-06 19:26:29 +05:30
if (isBotTurn)
{
currentPlayerTurnTimer?.KillTimer();
2026-02-06 19:26:29 +05:30
uIManager.UpdatePlayerTurnText(currentPlayerTypeTurn);
uIManager.UpdatePlayerTurnIcon(currentPlayerTypeTurn);
return;
2026-02-06 19:26:29 +05:30
}
}
// else
// SetCanRollDiceForUser(true); // TODO :: Need to change
2026-02-06 19:26:29 +05:30
2026-02-12 21:34:08 +05:30
uIManager.UpdatePlayerTurnText(currentPlayerTypeTurn, currentPlayerMaxTime);
uIManager.UpdatePlayerTurnIcon(currentPlayerTypeTurn);
if (currentPlayerTurnTimer == null)
currentPlayerTurnTimer = new TimerSystem();
2026-02-12 21:34:08 +05:30
currentPlayerTurnTimer.Init(currentPlayerMaxTime, onComplete: () =>
{
Debug.Log($"currentPlayerTurnTimer: HandleDiceViewForUser");
2026-02-12 21:34:08 +05:30
onComplete?.Invoke();
2026-02-06 19:26:29 +05:30
}, inProgress: (remTime) =>
{
if (HasGamePausedForUser())
{
savedCurrentPlayerTurnTimerInst = currentPlayerTurnTimer;
currentPlayerTurnTimer = null;
return;
}
2026-02-12 21:34:08 +05:30
uIManager.UpdatePlayerTurnText(currentPlayerTypeTurn, currentPlayerMaxTime - (int)remTime);
});
}
private void Update()
{
if (currentPlayerTurnTimer == null || !currentPlayerTurnTimer.IsInitialized)
{
return;
}
if (!currentPlayerTurnTimer.IsTimerComplete)
currentPlayerTurnTimer.UpdateTimer(Time.deltaTime);
else
ResetCurrentPlayerTurnTimer();
}
private void ResetCurrentPlayerTurnTimer()
{
currentPlayerTurnTimer?.KillTimer();
}
public void InitCurrentGamePlayerInfo()
2026-01-21 20:27:45 +05:30
{
currentPlayerTurnIndex = 0;
UpdateCurrentPlayerTurn(allPlayerTypes[currentPlayerTurnIndex]);
SetCanRollDiceForUser(IsUsersTurn());
2026-02-12 21:34:08 +05:30
UpdateDiceView();
Debug.Log($"currentPlayerTypeTurn: {currentPlayerTypeTurn}");
2026-02-04 12:09:04 +05:30
#if UNITY_EDITOR
SetCurrentSelectedPointer();
2026-02-04 12:09:04 +05:30
#endif
2026-01-21 20:27:45 +05:30
// initialize the board based on the player types
2026-01-30 22:06:51 +05:30
playerGameDatasDict = new Dictionary<PlayerType, PlayerGameData>();
2026-01-30 21:51:33 +05:30
2026-01-21 20:27:45 +05:30
foreach (PlayerGameData playerGameData in playerGameDatas)
{
2026-01-27 20:55:55 +05:30
if (!allPlayerTypes.Contains(playerGameData.playerType))
continue;
2026-01-21 20:27:45 +05:30
2026-01-27 20:55:55 +05:30
Debug.Log($"playerGameData.playerType: {playerGameData.playerType}");
2026-01-30 19:14:08 +05:30
2026-02-02 21:20:41 +05:30
playerGameDatasDict.Add(
playerGameData.playerType,
new PlayerGameData
{
playerType = playerGameData.playerType,
startIndex = playerGameData.startIndex,
endIndex = playerGameData.endIndex,
playersParent = playerGameData.playersParent,
playerPawnsDict = new Dictionary<int, PlayerPawn>(),
totalPawnsInHome = playerGameData.totalPawnsInHome,
totalPawnsFinished = playerGameData.totalPawnsFinished
});
foreach (Transform playerPawnChild in playerGameData.playersParent)
{
if (!playerPawnChild.gameObject.activeInHierarchy) continue;
2026-01-30 19:04:14 +05:30
2026-01-27 20:55:55 +05:30
var pawn = playerPawnChild.GetComponent<PlayerPawn>();
playerGameDatasDict[playerGameData.playerType].playerPawnsDict.Add(pawn.PlayerId, pawn);
2026-01-21 20:27:45 +05:30
}
2026-02-02 21:20:41 +05:30
2026-01-28 20:57:35 +05:30
playerGameDatasDict[playerGameData.playerType].totalPawnsInHome = playerGameDatasDict[playerGameData.playerType].playerPawnsDict.Count;
2026-01-21 20:27:45 +05:30
}
}
2026-02-12 21:34:08 +05:30
public void SetPlayerSelectionStates(bool state, Predicate<PlayerState> skipPredicate = null)
2026-01-21 20:27:45 +05:30
{
2026-01-23 12:46:59 +05:30
foreach (var playerPawn in playerGameDatasDict[currentPlayerTypeTurn].playerPawnsDict)
2026-01-21 20:27:45 +05:30
{
2026-02-12 21:34:08 +05:30
if (skipPredicate != null && skipPredicate.Invoke(playerPawn.Value.GetPlayerState()))
{
playerPawn.Value.SetPlayerSelectionState(false);
continue;
}
2026-01-21 20:27:45 +05:30
2026-01-27 20:55:55 +05:30
playerPawn.Value.SetPlayerSelectionState(state);
2026-01-21 20:27:45 +05:30
}
}
public void OnDiceInteracted()
2026-01-28 16:57:27 +05:30
{
2026-02-11 19:53:16 +05:30
RollDiceForUser();
}
2026-01-28 20:57:35 +05:30
public void RollDiceForPlayer(int rolledVal)
{
2026-01-28 16:57:27 +05:30
OnDiceRolled(rolledVal);
OnNoMovesLeft();
2026-01-28 16:57:27 +05:30
}
// Summary :: Function will be called and effective when a dice is rolled and no selection of pawns is made.
private void OnNoMovesLeft()
2026-01-28 16:57:27 +05:30
{
if (canSwitchPlayer && !CanRollDiceAgain)
{
Debug.Log($"Switching player");
SwitchPlayer();
}
}
private bool CheckForMaxDiceRollAttempt()
{
if (diceSixRollCounter > maxDiceSixRollCounter) // TODO :: Reset after test
{
CanRollDiceAgain = false;
SwitchPlayer();
return true;
}
return false;
2026-01-28 16:57:27 +05:30
}
private void RollDiceForBot(int rolledVal)
2026-01-28 16:57:27 +05:30
{
Debug.Log($"CallTest: RollDiceForBot");
2026-01-28 20:57:35 +05:30
// OnDiceRolled(diceValue == 0 ? UnityEngine.Random.Range(1, Ludo_3D_Constants.Max_Dice_Rolls + 1) : diceValue);
OnDiceRolled(rolledVal);
2026-01-28 16:57:27 +05:30
SelectPawnFromBotBase();
OnNoMovesLeft();
2026-01-28 16:57:27 +05:30
}
// TODO :: Call right after the dice is rolled with the animation
// What happens when you get a 6
private void SelectPawnFromBotBase()
{
2026-02-10 19:24:08 +05:30
if (canSwitchPlayer || availPlayersToMove.Count() < 1 && !CanRollDiceAgain || !botTypesInGame.Contains(currentPlayerTypeTurn))
2026-01-30 12:54:43 +05:30
{
Debug.Log($"returning from SelectPawnFromBotBase");
return; // Have a better check here
}
2026-01-28 16:57:27 +05:30
Debug.Log($"CallTest: SelectPawnFromBotBase: {currentPlayerTypeTurn}");
var botPawnsDictForCurrentPlayer = botRuntimeMovementData[currentPlayerTypeTurn]; // set the data inside this dict
2026-01-30 19:04:14 +05:30
2026-01-28 16:57:27 +05:30
int savedPlayerId = -1;
PlayerPawn pawn = null;
2026-02-10 19:24:08 +05:30
Debug.Log($"SelectPawnFromBotBase: availPlayers.Count(): {availPlayersToMove.Count()}, CanRollDiceAgain: {CanRollDiceAgain}");
2026-01-28 17:34:06 +05:30
2026-02-12 21:34:08 +05:30
InitActivePlayers();
#if UNITY_EDITOR
2026-01-28 20:57:35 +05:30
if (playerGameDatasDict[currentPlayerTypeTurn].totalPawnsInHome == 0)
{
2026-02-12 21:34:08 +05:30
SetDisplayCountForAllAvailPlayers(true);
2026-01-28 20:57:35 +05:30
}
2026-02-12 21:34:08 +05:30
#endif
2026-01-28 20:57:35 +05:30
if (CanRollDiceAgain) // got a 6 roll value
2026-01-28 16:57:27 +05:30
{
pawn = playerGameDatasDict[currentPlayerTypeTurn].playerPawnsDict.Values.FirstOrDefault(pawn => pawn.GetPlayerState() == PlayerState.InHome);
if (pawn != null)
{
Debug.Log($"SelectedPawn: {pawn.name}");
OnPawnSelected(pawn);
return;
}
2026-01-29 16:17:19 +05:30
}
2026-01-28 16:57:27 +05:30
2026-01-29 16:17:19 +05:30
Debug.Log($"Before Iterating");
2026-02-10 19:24:08 +05:30
foreach (var playerPawn in availPlayersToMove)
2026-01-28 16:57:27 +05:30
{
int possibleSteps = 0;
Tile possibleTileData = null;
2026-01-28 20:57:35 +05:30
if (playerPawn.GetPlayerState() != PlayerState.InFinishingPath)
FindPossibleTileData(playerPawn, out possibleSteps, out possibleTileData);
2026-01-28 20:57:35 +05:30
if (playerPawn.GetPlayerState() == PlayerState.InFinishingPath || possibleSteps > playerGameDatasDict[currentPlayerTypeTurn].endIndex) // TODO :: have a better second check
2026-01-28 16:57:27 +05:30
{
2026-01-29 16:17:19 +05:30
Debug.Log($"AI playerPawn :: {playerPawn.name} :: {playerPawn.PlayerId} :: inFinishingPath :: canSelectPlayer: {playerPawn.CanSelectPlayer}");
if (playerPawn.CanSelectPlayer)
{
botPawnsDictForCurrentPlayer[playerPawn.PlayerId] = BotMove.FinishingPathMove;
savedPlayerId = playerPawn.PlayerId;
break;
}
else
{
botPawnsDictForCurrentPlayer[playerPawn.PlayerId] = BotMove.NoMoves;
}
2026-01-28 16:57:27 +05:30
}
else if (possibleTileData != null && possibleTileData.IsSafeZone)
2026-01-28 16:57:27 +05:30
{
2026-01-29 16:17:19 +05:30
Debug.Log($"AI playerPawn :: {playerPawn.name} :: {playerPawn.PlayerId} :: safeMove");
2026-01-28 16:57:27 +05:30
botPawnsDictForCurrentPlayer[playerPawn.PlayerId] = BotMove.SafeMove;
}
else if (possibleTileData != null && possibleTileData.HasPawnsAvailable && possibleTileData.CurrentHoldingPlayerType != playerPawn.PlayerType)
2026-01-28 16:57:27 +05:30
{
2026-01-29 16:17:19 +05:30
Debug.Log($"AI playerPawn :: {playerPawn.name} :: {playerPawn.PlayerId} :: attackMove");
2026-01-28 16:57:27 +05:30
botPawnsDictForCurrentPlayer[playerPawn.PlayerId] = BotMove.AttackMove;
}
else
{
Debug.Log($"AI playerPawn :: {playerPawn.name} :: {playerPawn.PlayerId} :: normalMove");
2026-01-28 16:57:27 +05:30
botPawnsDictForCurrentPlayer[playerPawn.PlayerId] = BotMove.NormalMove;
}
}
List<int> playerIds = new List<int>();
if (savedPlayerId != -1)
Debug.Log($"SavedPlayerId: {savedPlayerId}, CanSelectPlayer: {playerGameDatasDict[currentPlayerTypeTurn].playerPawnsDict[savedPlayerId].CanSelectPlayer}");
if (savedPlayerId != -1 && playerGameDatasDict[currentPlayerTypeTurn].playerPawnsDict[savedPlayerId].CanSelectPlayer)
2026-01-28 16:57:27 +05:30
{
pawn = playerGameDatasDict[currentPlayerTypeTurn].playerPawnsDict[savedPlayerId]; // chances are when one of the character reaches towards the finishing point
}
else
{
if (botPawnsDictForCurrentPlayer.ContainsValue(BotMove.SafeMove) && botPawnsDictForCurrentPlayer.ContainsValue(BotMove.AttackMove))
{
InitPlayerIdsBasedOnState((val) => val == BotMove.AttackMove || val == BotMove.SafeMove);
savedPlayerId = playerIds[UnityEngine.Random.Range(0, playerIds.Count)];
pawn = playerGameDatasDict[currentPlayerTypeTurn].playerPawnsDict[savedPlayerId];
2026-01-28 20:57:35 +05:30
Debug.Log($"AI playerPawn :: {pawn.name} :: Getting savedplayer id");
2026-01-28 16:57:27 +05:30
}
else if (botPawnsDictForCurrentPlayer.ContainsValue(BotMove.SafeMove))
{
InitPawnBasedOnState(BotMove.SafeMove);
2026-01-28 20:57:35 +05:30
Debug.Log($"AI playerPawn :: {pawn.name} :: Getting savedplayer id");
2026-01-28 16:57:27 +05:30
}
else if (botPawnsDictForCurrentPlayer.ContainsValue(BotMove.AttackMove))
{
InitPawnBasedOnState(BotMove.AttackMove);
2026-01-28 20:57:35 +05:30
Debug.Log($"AI playerPawn :: {pawn.name} :: Getting savedplayer id");
2026-01-28 16:57:27 +05:30
}
else if (botPawnsDictForCurrentPlayer.ContainsValue(BotMove.NormalMove))
{
InitPawnBasedOnState(BotMove.NormalMove);
2026-01-28 20:57:35 +05:30
Debug.Log($"AI playerPawn :: {pawn.name} :: Getting savedplayer id");
2026-01-28 16:57:27 +05:30
}
}
if (pawn != null)
{
Debug.Log($"AI playerPawn :: {pawn.name} :: SelectedPawn: {pawn.name}");
OnPawnSelected(pawn);
}
else
{
if (CheckForMaxDiceRollAttempt())
{
return;
}
if (CanRollDiceAgain)
{
2026-02-11 19:53:16 +05:30
RollDiceForBot();
}
else
SwitchPlayer();
}
2026-01-28 16:57:27 +05:30
void InitPawnBasedOnState(BotMove botMove)
{
InitPlayerIdsBasedOnState((val) => val == botMove);
ReturnPlayerWithMaxSteps(ref savedPlayerId);
pawn = playerGameDatasDict[currentPlayerTypeTurn].playerPawnsDict[savedPlayerId];
}
void InitPlayerIdsBasedOnState(Predicate<BotMove> state)
{
foreach (var botPair in botPawnsDictForCurrentPlayer)
if (state.Invoke(botPair.Value))
playerIds.Add(botPair.Key);
}
void ReturnPlayerWithMaxSteps(ref int savedPlayerId)
{
int maxStepsTaken = -999;
foreach (var id in playerIds)
{
if (playerGameDatasDict[currentPlayerTypeTurn].playerPawnsDict[id].StepsTaken > maxStepsTaken)
{
maxStepsTaken = playerGameDatasDict[currentPlayerTypeTurn].playerPawnsDict[id].StepsTaken;
savedPlayerId = id;
}
}
}
}
private void FindPossibleTileData(PlayerPawn playerPawn, out int possibleSteps, out Tile possibleTileData)
{
2026-02-02 20:12:00 +05:30
int lastStepGenTile = totalStepsForCharacter - TilesManager.GetFinishingTileDataLength(playerPawn.PlayerType);//playerGameDatasDict[playerPawn.PlayerType].endIndex;
int possibleTileIndex = playerPawn.CurrentTileIndex + diceRolledValue;
2026-02-02 20:12:00 +05:30
int lastTileIndex = TilesManager.GetGeneralTilesLength() - 1;
bool canGoingInsideFinishingPath = IsGoingInsideFinishingPath(playerPawn, out possibleSteps);
int index = canGoingInsideFinishingPath ? possibleSteps - lastStepGenTile - 1
: possibleTileIndex > lastTileIndex ? possibleTileIndex - lastTileIndex - 1 : possibleTileIndex; // case for addressing the scenario when going through the last index of general tiles.
Debug.Log($"possibleTileIndex: {possibleTileIndex}, lastStepGenTileIdx: {lastStepGenTile}");
Debug.Log($"index: {index}");
2026-02-02 20:12:00 +05:30
possibleTileData = canGoingInsideFinishingPath ? TilesManager.RetrieveFinishingTileBasedOnIndex(playerPawn.PlayerType, index) : TilesManager.RetrieveTileBasedOnIndex(index);
}
public bool IsGoingInsideFinishingPath(PlayerPawn playerPawn, out int possibleSteps)
{
possibleSteps = playerGameDatasDict[playerPawn.PlayerType].playerPawnsDict[playerPawn.PlayerId].StepsTaken + diceRolledValue;
2026-02-02 20:12:00 +05:30
return possibleSteps > TilesManager.GetGeneralTilesLength() - 1;
}
2026-02-11 19:53:16 +05:30
private bool HasNoPlayersTravelling() => Mathf.Abs(availPlayersToMove.Count - playerGameDatasDict[currentPlayerTypeTurn].totalPawnsInFinishingPath) < 1;
2026-02-12 21:34:08 +05:30
private bool IsUsersTurn() => gameModeHandler.CurrentGameModeType != GameModeType.Bot ||
!botTypesInGame.Contains(currentPlayerTypeTurn);
2026-02-11 19:53:16 +05:30
2026-01-21 20:27:45 +05:30
public void OnDiceRolled(int rolledVal)
{
2026-01-29 22:16:54 +05:30
SetCanRollDiceForUser(false);
2026-01-21 20:27:45 +05:30
// add core dice logic here
Debug.Log($"Tile Index :: LUDO :: rolledVal: {rolledVal} :: {currentPlayerTypeTurn}");
2026-01-21 20:27:45 +05:30
diceRolledValue = rolledVal;
2026-01-26 19:48:55 +05:30
diceText.text = $"{diceRolledValue}";
2026-02-11 19:53:16 +05:30
2026-02-12 21:34:08 +05:30
if (!CanRollDiceAgain) // remove this check for showing arrow logic
{
2026-02-12 21:34:08 +05:30
#if UNITY_EDITOR
2026-02-10 19:24:08 +05:30
SetDisplayCountForAllAvailPlayers(true);
2026-02-12 21:34:08 +05:30
#endif
}
FilterAvailablePlayersToMove();
2026-01-21 20:27:45 +05:30
if (rolledVal == Ludo_3D_Constants.Max_Dice_Rolls)
{
2026-01-28 16:57:27 +05:30
canSwitchPlayer = false;
2026-01-21 20:27:45 +05:30
// provide option to select a pawn from the list
// also play a simple animation before selecting
CanRollDiceAgain = true;
diceSixRollCounter++;
2026-02-10 19:24:08 +05:30
2026-02-12 21:34:08 +05:30
if (IsUsersTurn())
2026-02-10 19:24:08 +05:30
{
2026-02-12 21:34:08 +05:30
Debug.Log($"availPlayersToMove.Count < 1: {availPlayersToMove.Count < 1} || HasNoPlayersTravelling(): {HasNoPlayersTravelling()}");
2026-02-11 19:53:16 +05:30
if (availPlayersToMove.Count < 1 || HasNoPlayersTravelling())
2026-02-10 19:24:08 +05:30
{
if (playerGameDatasDict[currentPlayerTypeTurn].totalPawnsInHome > 0)
{
OnPawnSelected(playerGameDatasDict[currentPlayerTypeTurn].playerPawnsDict.Values
.FirstOrDefault(pawn => pawn.GetPlayerState() == PlayerState.InHome));
return;
}
}
else
{
if (playerGameDatasDict[currentPlayerTypeTurn].totalPawnsInHome < 1 && CanMoveSoloPlayer())
{
SetPlayerSelectionStates(false);
OnPawnSelected(availPlayersToMove[0]);
2026-02-10 19:24:08 +05:30
return;
}
2026-02-10 19:24:08 +05:30
}
}
2026-02-04 12:09:04 +05:30
#if UNITY_EDITOR
pointerMeshRend.material = selectMat;
2026-02-04 12:09:04 +05:30
#endif
Debug.Log($"### AreAllPawnsInFinishingPath");
2026-01-27 20:55:55 +05:30
if (AreAllPawnsInFinishingPath())
{
2026-02-12 21:34:08 +05:30
SetCanRollDiceForUser(IsUsersTurn());
UpdateResponseTimerForUser(currentPlayerSelectionMaxTime, () => RollDiceForUser());
CheckForMaxDiceRollAttempt();
return;
}
2026-02-12 21:34:08 +05:30
SetPlayerSelectionStates(true, (state) => state == PlayerState.InFinishingPath || state == PlayerState.HasFinished);
2026-01-21 20:27:45 +05:30
}
2026-02-12 21:34:08 +05:30
else
2026-01-21 20:27:45 +05:30
{
2026-02-10 19:24:08 +05:30
Debug.Log($"before CustomAvailablePlayers: {availPlayersToMove.Count}");
2026-02-10 19:24:08 +05:30
Debug.Log($"CustomAvailablePlayers: {availPlayersToMove.Count}");
canSwitchPlayer = availPlayersToMove.Count < 1;
CanRollDiceAgain = false;
}
diceRollHandler.DiceView.SetDiceButtonInteraction(CanRollDiceAgain);
Debug.Log($"CanRollDiceAgain: {CanRollDiceAgain}, canSwitchPlayer: {canSwitchPlayer}");
if (IsUsersTurn())
{
Action onComplete = null;
if (CanRollDiceAgain && availPlayersToMove.Count < 1 &&
playerGameDatasDict[currentPlayerTypeTurn].totalPawnsInHome > 0)
2026-02-10 19:24:08 +05:30
{
onComplete = () => OnPawnSelected(playerGameDatasDict[currentPlayerTypeTurn].playerPawnsDict.Values
.FirstOrDefault(pawn => pawn.GetPlayerState() == PlayerState.InHome));
UpdateResponseTimerForUser(currentPlayerSelectionMaxTime, () => onComplete?.Invoke());
}
else if (availPlayersToMove.Count > 0)
{
if (CanRollDiceAgain && playerGameDatasDict[currentPlayerTypeTurn].totalPawnsInHome > 0)
{
onComplete = () => OnPawnSelected(playerGameDatasDict[currentPlayerTypeTurn].playerPawnsDict.Values
.FirstOrDefault(pawn => pawn.GetPlayerState() == PlayerState.InHome));
UpdateResponseTimerForUser(currentPlayerSelectionMaxTime, () => onComplete?.Invoke());
return;
}
2026-02-10 19:24:08 +05:30
if (CanMoveSoloPlayer())
{
2026-02-12 21:34:08 +05:30
SetPlayerSelectionStates(false);
2026-02-10 19:24:08 +05:30
OnPawnSelected(availPlayersToMove[0]);
return;
2026-02-10 19:24:08 +05:30
}
onComplete = () => OnPawnSelected(availPlayersToMove.OrderByDescending(pawn => pawn.StepsTaken).FirstOrDefault());
UpdateResponseTimerForUser(currentPlayerSelectionMaxTime, () => onComplete?.Invoke());
2026-02-10 19:24:08 +05:30
}
2026-01-21 20:27:45 +05:30
}
}
2026-01-28 16:57:27 +05:30
private void FilterAvailablePlayersToMove()
{
List<int> indexesToRemove = new List<int>();
for (int i = 0; i < availPlayersToMove.Count; i++)
2026-02-12 21:34:08 +05:30
{
Debug.Log($"## playerPawn.GetPlayerState(): {availPlayersToMove[i].GetPlayerState()}");
if (availPlayersToMove[i].GetPlayerState() == PlayerState.InFinishingPath)
{
Debug.Log($"diceRolledValue: {diceRolledValue}, FinishingDataLen: {TilesManager.GetFinishingTileDataLength(currentPlayerTypeTurn)}, playerPawn.CurrentTileIndex: {availPlayersToMove[i].CurrentTileIndex}");
if (diceRolledValue <= TilesManager.GetFinishingTileDataLength(currentPlayerTypeTurn) - (availPlayersToMove[i].CurrentTileIndex + 1))
{
availPlayersToMove[i].SetPlayerSelectionState(true);
}
else
{
indexesToRemove.Add(i);
availPlayersToMove[i].SetPlayerSelectionState(false);
#if UNITY_EDITOR
availPlayersToMove[i].ShowPlayerCountCanvas(false);
#endif
}
continue;
}
availPlayersToMove[i].SetPlayerSelectionState(true);
2026-02-12 21:34:08 +05:30
}
for (int idx = indexesToRemove.Count - 1; idx >= 0; idx--)
availPlayersToMove.RemoveAt(idx);
2026-01-21 20:27:45 +05:30
}
2026-02-10 19:24:08 +05:30
private bool CanMoveSoloPlayer()
{
if (availPlayersToMove.Count == 1)
{
if (availPlayersToMove[0].GetPlayerState() == PlayerState.InSafeZone || availPlayersToMove[0].GetPlayerState() == PlayerState.Moving
|| availPlayersToMove[0].GetPlayerState() == PlayerState.InFinishingPath)
{
return true;
}
}
return false;
}
private void InitActivePlayers()
{
2026-02-11 19:53:16 +05:30
availPlayersToMove = new List<PlayerPawn>();
2026-02-10 19:24:08 +05:30
availPlayersToMove = playerGameDatasDict[currentPlayerTypeTurn].playerPawnsDict.Values.Select(pawn => pawn)
.Where(pawn => pawn.GetPlayerState() == PlayerState.InSafeZone ||
pawn.GetPlayerState() == PlayerState.Moving ||
pawn.GetPlayerState() == PlayerState.InFinishingPath).ToList();
}
2026-01-27 20:55:55 +05:30
private bool AreAllPawnsInFinishingPath()
{
bool areAllPawnsInFinishingPath = false;
foreach (var pawn in playerGameDatasDict[currentPlayerTypeTurn].playerPawnsDict)
{
if (pawn.Value.GetPlayerState() == PlayerState.HasFinished) continue;
if (pawn.Value.GetPlayerState() == PlayerState.InFinishingPath)
2026-01-27 20:55:55 +05:30
{
areAllPawnsInFinishingPath = true;
continue;
}
areAllPawnsInFinishingPath = false;
break;
}
return areAllPawnsInFinishingPath;
}
2026-01-28 16:57:27 +05:30
private void UpdatePlayerState(PlayerPawn playerPawn, PlayerState playerState)
{
if (!playerPawn)
{
// Debug.LogError($"Player pawn is null");
return;
}
2026-01-30 19:04:14 +05:30
Debug.Log($"#### UpdatePlayerState ");
2026-01-28 16:57:27 +05:30
playerGameDatasDict[playerPawn.PlayerType].playerPawnsDict[playerPawn.PlayerId] = playerPawn;
playerGameDatasDict[playerPawn.PlayerType].playerPawnsDict[playerPawn.PlayerId].SetPlayerState(playerState);
playerPawn.SetPlayerState(playerState);
}
private void CheckDiceRollForBot()
2026-01-28 17:34:06 +05:30
{
if (CanRollDiceAgain)
{
2026-02-11 19:53:16 +05:30
RollDiceForBot();
}
}
2026-02-11 19:53:16 +05:30
private void RollDiceForUser()
{
ResetCurrentPlayerTurnTimer();
2026-02-11 19:53:16 +05:30
InitActivePlayers();
bool canUsePawnsFromHome = CanUsePawnsFromHome();
diceRollHandler.HandleDiceViewForUser(
hasNoActionOnRoll: canUsePawnsFromHome,
playerBase: canUsePawnsFromHome ? PlayerBaseHandler.GetPlayerBase(currentPlayerTypeTurn) : null);
}
private void RollDiceForBot()
{
ResetCurrentPlayerTurnTimer();
2026-02-11 19:53:16 +05:30
InitActivePlayers();
HandleDiceRollForBot();
}
/***
* Summary:
* Scenario where the user/bot has no other move when some of the pawns/characters out of home
* are in finishing path and doesn't have a move from dice rolled value
*/
private bool CanUsePawnsFromHome()
{
return playerGameDatasDict[currentPlayerTypeTurn].totalPawnsInHome > 0 && HasNoPlayersTravelling();
}
private void HandleDiceRollForBot()
{
2026-02-04 12:32:18 +05:30
if (isDebugAITest)
2026-02-11 19:53:16 +05:30
diceRollHandler.HandleDiceViewForBot((rollVal) => RollDiceForBot(rollVal),
diceValue == 0 ? UnityEngine.Random.Range(1, Ludo_3D_Constants.Max_Dice_Rolls + 1) : diceValue);
else
{
bool canUsePawnsFromHome = CanUsePawnsFromHome();
diceRollHandler.HandleDiceViewForBot(
(rollVal) => RollDiceForBot(rollVal),
hasNoActionOnRoll: canUsePawnsFromHome,
playerBase: canUsePawnsFromHome ? PlayerBaseHandler.GetPlayerBase(currentPlayerTypeTurn) : null);
}
2026-01-28 17:34:06 +05:30
}
2026-02-02 15:10:51 +05:30
public void OnPawnSelected(PlayerPawn selectedPawn)
2026-01-21 20:27:45 +05:30
{
2026-02-12 21:34:08 +05:30
// TODO :: Hide the selectable characters indicator here
SetPlayerSelectionStates(false);
if (IsUsersTurn()) currentPlayerTurnTimer.KillTimer();
2026-01-21 20:27:45 +05:30
PlayerGameData playerGameData = playerGameDatasDict[currentPlayerTypeTurn];
2026-02-02 15:10:51 +05:30
Debug.Log($"playerPawn.GetPlayerState(): {selectedPawn.GetPlayerState()}");
2026-01-21 20:27:45 +05:30
2026-02-12 21:34:08 +05:30
#if UNITY_EDITOR
selectedPawn.ShowPlayerCountCanvas(false);
#endif
2026-02-02 15:10:51 +05:30
if (selectedPawn.GetPlayerState() == PlayerState.InHome)
2026-01-21 20:27:45 +05:30
{
2026-02-02 20:12:00 +05:30
Tile targetTile = TilesManager.RetrieveTileBasedOnIndex(playerGameData.startIndex);
2026-01-23 12:46:59 +05:30
2026-02-02 15:10:51 +05:30
selectedPawn.MoveToTile(
2026-02-02 20:12:00 +05:30
TilesManager.GetAndInitPositionInsideSafeZone(selectedPawn, targetTile, currentPlayerTypeTurn),
2026-01-21 20:27:45 +05:30
onComplete: () =>
{
2026-01-28 20:57:35 +05:30
playerGameDatasDict[playerGameData.playerType].totalPawnsInHome--;
2026-02-02 15:10:51 +05:30
UpdatePlayerState(selectedPawn, PlayerState.InSafeZone);
2026-02-12 21:34:08 +05:30
#if UNITY_EDITOR
ShowUpdatedPlayerCountOnTile(selectedPawn);
2026-02-12 21:34:08 +05:30
#endif
if (CheckForMaxDiceRollAttempt())
{
return;
}
2026-02-12 21:34:08 +05:30
2026-02-02 15:10:51 +05:30
if (selectedPawn.IsBotPlayer)
CheckDiceRollForBot();
else
{
SetCanRollDiceForUser(IsUsersTurn());
2026-02-12 21:34:08 +05:30
UpdateResponseTimerForUser(currentPlayerMaxTime: currentPlayerTurnMaxTime, () => RollDiceForUser());
}
2026-01-21 20:27:45 +05:30
}, playerGameData.startIndex);
return;
}
2026-02-02 15:10:51 +05:30
else if (selectedPawn.GetPlayerState() == PlayerState.InFinishingPath)
{
2026-02-02 20:12:00 +05:30
Tile currentSittingTile = TilesManager.RetrieveFinishingTileBasedOnIndex(selectedPawn.PlayerType, selectedPawn.CurrentTileIndex);
currentSittingTile.ResetPlayerPawn(selectedPawn);
2026-02-12 21:34:08 +05:30
#if UNITY_EDITOR
if (currentSittingTile.HasPawnsAvailable)
{
var playerPawns = currentSittingTile.GetPlayerPawns();
foreach (var pawn in playerPawns)
ShowUpdatedPlayerCountOnTile(pawn);
}
2026-02-12 21:34:08 +05:30
#endif
2026-02-02 15:10:51 +05:30
ApplyFinishingPathLogic(selectedPawn);
}
2026-02-02 15:10:51 +05:30
else if (selectedPawn.CurrentTileIndex == playerGameDatasDict[currentPlayerTypeTurn].endIndex)
{
2026-02-02 20:12:00 +05:30
TilesManager.RetrieveTileBasedOnIndex(selectedPawn.CurrentTileIndex).ResetPlayerPawn(selectedPawn);
2026-02-10 19:24:08 +05:30
playerGameDatasDict[currentPlayerTypeTurn].totalPawnsInFinishingPath++;
2026-02-02 15:10:51 +05:30
ApplyFinishingPathLogic(selectedPawn);
}
2026-02-02 15:10:51 +05:30
else if (selectedPawn.GetPlayerState() == PlayerState.InSafeZone || selectedPawn.GetPlayerState() == PlayerState.Moving)
2026-01-21 20:27:45 +05:30
{
// move based on the dice value
2026-02-02 15:10:51 +05:30
Debug.Log($"Tile Index :: currentTileIndex: {selectedPawn.CurrentTileIndex}");
2026-02-02 20:12:00 +05:30
int nextTileIdx = TilesManager.GetNextGeneralTileIndex(selectedPawn.CurrentTileIndex);
2026-02-02 15:10:51 +05:30
int targetIdx = selectedPawn.CurrentTileIndex + diceRolledValue;
2026-01-21 20:27:45 +05:30
if (nextTileIdx == 0)
2026-02-02 15:10:51 +05:30
targetIdx = (targetIdx - selectedPawn.CurrentTileIndex) - 1;
2026-01-30 19:04:14 +05:30
2026-02-02 20:12:00 +05:30
Tile currentSittingTile = TilesManager.RetrieveTileBasedOnIndex(selectedPawn.CurrentTileIndex);
if (currentSittingTile.IsSafeZone)
{
2026-02-04 12:32:18 +05:30
SafeTile currentSittingSafeTile = (SafeTile)currentSittingTile;
currentSittingSafeTile.UpdateSafeZonePlayerData(currentPlayerTypeTurn, selectedPawn);
2026-02-04 12:32:18 +05:30
if (currentSittingSafeTile.PlayerTypesCount == 1) // && currentSittingSafeTile.ContainsPlayerType(selectedPawn.PlayerType))
{
var playerPawnsTest = currentSittingSafeTile.GetFirstPlayerPawns();
foreach (var pawn in playerPawnsTest)
pawn.MoveToCustomTilePosition(currentSittingSafeTile.CenterPlacementPosition);
2026-01-23 13:33:16 +05:30
}
2026-02-12 21:34:08 +05:30
#if UNITY_EDITOR
if (currentSittingSafeTile.ContainsPlayerType(currentPlayerTypeTurn))
{
var playerPawns = currentSittingSafeTile.GetPlayerPawns(currentPlayerTypeTurn);
foreach (var pawn in playerPawns)
ShowUpdatedPlayerCountOnTile(pawn);
}
2026-02-12 21:34:08 +05:30
#endif
}
2026-01-26 13:03:50 +05:30
else
{
2026-02-02 15:10:51 +05:30
currentSittingTile.ResetPlayerPawn(selectedPawn);
2026-02-12 21:34:08 +05:30
#if UNITY_EDITOR
2026-02-02 15:10:51 +05:30
if (currentSittingTile.HasPawnsAvailable)
{
var playerPawns = currentSittingTile.GetPlayerPawns();
foreach (var pawn in playerPawns)
ShowUpdatedPlayerCountOnTile(pawn);
2026-02-02 15:10:51 +05:30
}
2026-02-12 21:34:08 +05:30
#endif
2026-01-26 13:03:50 +05:30
}
2026-02-02 15:10:51 +05:30
MoveThroughTiles(selectedPawn, nextTileIdx, targetIndex: targetIdx);
2026-01-21 20:27:45 +05:30
}
}
private void ApplyFinishingPathLogic(PlayerPawn playerPawn)
{
2026-02-02 20:12:00 +05:30
int finishingPathIndex = TilesManager.GetNextFinishingTileIndex(playerPawn.CurrentTileIndex, playerPawn.PlayerType);
int targetIdx = finishingPathIndex + diceRolledValue > TilesManager.GetFinishingTileDataLength(currentPlayerTypeTurn) - 1 ?
TilesManager.GetFinishingTileDataLength(currentPlayerTypeTurn) - 1 : finishingPathIndex + diceRolledValue;
Debug.Log($"TargetIdx: {targetIdx}, finishingPathIndex: {finishingPathIndex}");
MoveThroughFinishingPath(playerPawn, finishingPathIndex, targetIdx);
}
2026-02-02 20:12:00 +05:30
// TODO :: move to tiles manager
private void SwitchPlayer(PlayerPawn playerPawn = null)
2026-01-21 20:27:45 +05:30
{
2026-01-28 16:57:27 +05:30
Debug.Log($"CallTest: SwitchPlayer");
2026-01-30 19:04:14 +05:30
if (playerPawn)
2026-02-02 20:12:00 +05:30
UpdatePlayerState(playerPawn, TilesManager.RetrieveTileBasedOnIndex(playerPawn.CurrentTileIndex).IsSafeZone ? PlayerState.InSafeZone : PlayerState.Moving);
2026-01-28 16:57:27 +05:30
if (!CanRollDiceAgain)
2026-01-21 20:27:45 +05:30
{
Debug.Log($"currentPlayerTurn: {currentPlayerTypeTurn}");
Debug.Log($"currentPlayerTurnIndex: {currentPlayerTurnIndex}");
2026-02-10 19:24:08 +05:30
Debug.Log($"before SwitchPlayer availPlayers: {availPlayersToMove.Count}, playerPawn: {playerPawn}");
2026-02-12 21:34:08 +05:30
InitActivePlayers();
#if UNITY_EDITOR
SetDisplayCountForAllAvailPlayers(false);
#endif
2026-02-10 19:24:08 +05:30
Debug.Log($"after SwitchPlayer availPlayers: {availPlayersToMove.Count}, playerPawn: {playerPawn}");
2026-02-02 21:20:41 +05:30
Debug.Log($"after allPlayerTypes.Count: {allPlayerTypes.Count}");
2026-01-30 19:04:14 +05:30
2026-02-02 20:12:00 +05:30
if (allPlayerTypes.Count == 0)
{
Debug.LogError($"GAME IS OVER");
return;
}
currentPlayerTurnIndex = allPlayerTypes.FindIndex(type => type == currentPlayerTypeTurn);
2026-01-27 20:55:55 +05:30
if (currentPlayerTypeTurn == allPlayerTypes[allPlayerTypes.Count - 1])
2026-01-21 20:27:45 +05:30
{
currentPlayerTurnIndex = 0;
}
else
{
currentPlayerTurnIndex++;
}
soundManager.PlayGameSoundClip(SoundType.Turn);
UpdateCurrentPlayerTurn(allPlayerTypes[currentPlayerTurnIndex]);
2026-02-12 21:34:08 +05:30
Debug.Log($"IsUsersTurn: {IsUsersTurn()}");
if (IsUsersTurn())
{
SetCanRollDiceForUser(true);
2026-02-12 21:34:08 +05:30
Invoke(nameof(UpdateDiceView), diceRollDelayForUser);
}
else
{
SetCanRollDiceForUser(false);
currentPlayerTurnTimer?.KillTimer();
2026-02-12 21:34:08 +05:30
Invoke(nameof(UpdateDiceViewForBot), diceRollDelayForBot);
}
2026-01-30 22:17:18 +05:30
diceSixRollCounter = 0;
2026-01-26 19:48:55 +05:30
diceText.text = $"{0}";
2026-01-21 20:27:45 +05:30
}
// else
// {
// UpdateResponseTimerForUser(currentPlayerMaxTime: currentPlayerTurnMaxTime, () => RollDiceForUser());
//
// if (gameModeHandler.CurrentGameModeType == GameModeType.Bot && botTypesInGame.Contains(currentPlayerTypeTurn)) // TODO :: Double check calling of the function
// {
// Debug.Log($"Invoking RollDiceForBot");
// HandleDiceRollWithDelay();
// }
// }
2026-02-02 21:20:41 +05:30
Debug.Log($"currentPlayerTurnIndex: {currentPlayerTurnIndex}");
2026-01-26 13:03:50 +05:30
Debug.Log($"CurrentPlayerTurn: {currentPlayerTypeTurn}");
2026-01-21 20:27:45 +05:30
2026-02-02 20:12:00 +05:30
#if UNITY_EDITOR
SetCurrentSelectedPointer();
2026-01-21 20:27:45 +05:30
pointerMeshRend.material = turnMat;
// pointerMeshRend.materials[0] = turnMat;
2026-02-02 20:12:00 +05:30
#endif
}
private void OnCanRollDiceAgain(PlayerPawn playerPawn = null)
{
if (playerPawn)
UpdatePlayerState(playerPawn, TilesManager.RetrieveTileBasedOnIndex(playerPawn.CurrentTileIndex).IsSafeZone ? PlayerState.InSafeZone : PlayerState.Moving);
UpdateResponseTimerForUser(currentPlayerMaxTime: currentPlayerTurnMaxTime, () => RollDiceForUser());
if (gameModeHandler.CurrentGameModeType == GameModeType.Bot && botTypesInGame.Contains(currentPlayerTypeTurn)) // TODO :: Double check calling of the function
{
Debug.Log($"Invoking RollDiceForBot");
HandleDiceRollWithDelay();
}
}
2026-02-12 21:34:08 +05:30
2026-01-21 20:27:45 +05:30
private void MoveThroughTiles(PlayerPawn playerPawn, int index, int targetIndex)
{
2026-02-02 20:12:00 +05:30
Tile nextTile = TilesManager.RetrieveTileBasedOnIndex(index);
2026-01-23 12:46:59 +05:30
Vector3 targetPosition = nextTile.CenterPlacementPosition;
2026-01-23 12:46:59 +05:30
Debug.Log($"Tile Index :: nextIndex: {index}, targetIndex: {targetIndex}, nextTileName: {nextTile.name}");
if (index == targetIndex) // if the target index is the safe zone only then apply the logic for rearranging pawns
{
2026-02-02 20:12:00 +05:30
Tile targetTile = TilesManager.RetrieveTileBasedOnIndex(targetIndex);
if (targetTile.IsSafeZone)
2026-01-21 20:27:45 +05:30
{
2026-02-02 20:12:00 +05:30
targetPosition = TilesManager.GetAndInitPositionInsideSafeZone(playerPawn, targetTile, currentPlayerTypeTurn);
}
}
2026-01-23 12:46:59 +05:30
Debug.Log($"tile targetPosition: {targetPosition}");
2026-02-12 21:34:08 +05:30
#if UNITY_EDITOR
playerPawn.ShowPlayerCountCanvas(false); // TODO :: Check if call can be removed
#endif
2026-01-23 12:46:59 +05:30
playerPawn.MoveToTile(
targetPosition,
onComplete: () =>
{
diceRolledValue--;
2026-01-23 12:46:59 +05:30
Debug.Log($"DiceRolledValue: {diceRolledValue}");
if (diceRolledValue > 0)
{
2026-02-02 20:12:00 +05:30
int nextTileIndex = TilesManager.GetNextGeneralTileIndex(playerPawn.CurrentTileIndex);
2026-01-23 12:46:59 +05:30
Debug.Log($"currentTileIndex: {playerPawn.CurrentTileIndex}, nextTileIndex: {nextTileIndex}, targetIndex: {targetIndex}");
if (playerPawn.GetPlayerState() == PlayerState.InFinishingPath || index == playerGameDatasDict[currentPlayerTypeTurn].endIndex)
2026-01-21 20:27:45 +05:30
{
// MoveThroughTiles(playerPawn, index, targetIndex);
Debug.Log($"TargetIdx: {targetIndex - index}");
2026-02-10 19:24:08 +05:30
if (index == playerGameDatasDict[currentPlayerTypeTurn].endIndex)
playerGameDatasDict[currentPlayerTypeTurn].totalPawnsInFinishingPath++;
CheckForGamePause(() => MoveThroughFinishingPath(playerPawn, 0, targetIndex - index));
2026-01-21 20:27:45 +05:30
}
else if (nextTileIndex <= targetIndex)
{
if (nextTileIndex == 0)
2026-01-23 12:46:59 +05:30
targetIndex = (targetIndex - playerPawn.CurrentTileIndex) - 1;
2026-01-21 20:27:45 +05:30
CheckForGamePause(() => MoveThroughTiles(playerPawn, nextTileIndex, targetIndex));
}
}
else
{
2026-01-30 19:04:14 +05:30
// TODO :: Improve this logic, use a collection
2026-01-26 13:03:50 +05:30
Debug.Log($"nextTile.IsSafeZone: {nextTile.IsSafeZone}");
2026-01-30 19:04:14 +05:30
if (CanRollDiceAgain)
2026-02-12 21:34:08 +05:30
{
#if UNITY_EDITOR
ShowUpdatedPlayerCountOnTile(playerPawn);
2026-02-12 21:34:08 +05:30
#endif
}
2026-01-30 19:04:14 +05:30
2026-01-23 12:46:59 +05:30
if (!nextTile.IsSafeZone)
{
Debug.Log($"nextTile.HasPawnsAvailable: {nextTile.HasPawnsAvailable}");
if (nextTile.HasPawnsAvailable && playerPawn.PlayerType != nextTile.CurrentHoldingPlayerType)
2026-01-23 12:46:59 +05:30
{
Debug.Log($"nextTile.PlayerPawn: {nextTile.CurrentHoldingPlayerType}, {nextTile.transform.name}");
Debug.Log($"nextTile.TotalPawnsInTile: {nextTile.TotalPawnsInTile}");
2026-01-26 13:03:50 +05:30
// play animation for moving back to base.
2026-01-23 12:46:59 +05:30
// TODO :: Send existing pawn back to base.
// means there's already a pawn there, move him back to the base.
int counter = nextTile.TotalPawnsInTile;
for (int i = counter; i > 0; i--)
{
var pawn = nextTile.GetPlayerPawn();
playerGameDatasDict[pawn.PlayerType].totalPawnsInHome++;
2026-02-02 19:27:17 +05:30
playerBaseHandler.SendPlayerToHome(pawn);
}
if (CheckForMaxDiceRollAttempt())
{
nextTile.InitPlayerPawn(playerPawn, currentPlayerTypeTurn);
2026-02-02 15:10:51 +05:30
// UpdatePlayerCountOnTile(playerPawn, true);
return;
}
CanRollDiceAgain = true;
2026-01-23 12:46:59 +05:30
}
nextTile.InitPlayerPawn(playerPawn, currentPlayerTypeTurn);
2026-02-12 21:34:08 +05:30
#if UNITY_EDITOR
ShowUpdatedPlayerCountOnTile(playerPawn);
2026-02-12 21:34:08 +05:30
#endif
if (CheckForMaxDiceRollAttempt())
{
return;
}
}
else
{
if (CheckForMaxDiceRollAttempt())
{
return;
}
2026-01-23 12:46:59 +05:30
}
if (!CanRollDiceAgain) SwitchPlayer(playerPawn);
else
{
if (IsUsersTurn())
{
UpdateResponseTimerForUser(currentPlayerMaxTime: currentPlayerTurnMaxTime, () => RollDiceForUser());
SetCanRollDiceForUser(IsUsersTurn());
}
2026-01-23 12:46:59 +05:30
OnCanRollDiceAgain(playerPawn);
diceRollHandler.DiceView.SetDiceButtonInteraction(true);
}
2026-01-21 20:27:45 +05:30
}
2026-01-30 19:04:14 +05:30
},
2026-01-21 20:27:45 +05:30
index);
}
private bool HasGamePausedForUser()
{
return (botTypesInGame == null || !botTypesInGame.Contains(currentPlayerTypeTurn)) &&
GameManager.CurrentGameState == GameState.IsPaused;
}
public void CheckForGamePause(Action onComplete)
{
Debug.Log($"CheckForGamePause: {GameManager.CurrentGameState == GameState.IsPaused}");
if (HasGamePausedForUser())
{
onGameResumed = onComplete;
}
else
{
onComplete?.Invoke();
}
}
public void OnGameResumed()
{
onGameResumed?.Invoke();
currentPlayerTurnTimer = savedCurrentPlayerTurnTimerInst;
savedCurrentPlayerTurnTimerInst = null;
}
2026-01-21 20:27:45 +05:30
private void MoveThroughFinishingPath(PlayerPawn playerPawn, int index, int targetIndex)
{
2026-01-28 16:57:27 +05:30
UpdatePlayerState(playerPawn, PlayerState.InFinishingPath);
2026-01-30 19:14:08 +05:30
2026-01-21 20:27:45 +05:30
playerPawn.MoveToTile(
2026-02-02 20:12:00 +05:30
TilesManager.RetrieveFinishingTileBasedOnIndex(currentPlayerTypeTurn, index).transform.position,
2026-01-21 20:27:45 +05:30
onComplete: () =>
{
diceRolledValue--;
Debug.Log($"DiceRolledValue: {diceRolledValue}");
if (diceRolledValue > 0)
2026-01-21 20:27:45 +05:30
{
2026-02-02 20:12:00 +05:30
int tileIndex = TilesManager.GetNextFinishingTileIndex(playerPawn.CurrentTileIndex, playerPawn.PlayerType);
Debug.Log($"tileIndex: {tileIndex}, targetIndex: {targetIndex}");
if (tileIndex <= targetIndex)
{
CheckForGamePause(() => MoveThroughFinishingPath(playerPawn, tileIndex, targetIndex));
}
2026-01-21 20:27:45 +05:30
}
else
{
2026-02-02 20:12:00 +05:30
if (playerPawn.CurrentTileIndex == TilesManager.GetFinishingTileDataLength(currentPlayerTypeTurn) - 1)
{
onGameResumed = null;
2026-02-02 20:12:00 +05:30
Tile tile = TilesManager.RetrieveFinishingTileBasedOnIndex(playerPawn.PlayerType, playerPawn.CurrentTileIndex);
2026-02-02 19:27:17 +05:30
tile.InitPlayerPawn(playerPawn, playerPawn.PlayerType);
// ShowUpdatedPlayerCountOnTile(playerPawn);
2026-02-02 19:27:17 +05:30
2026-01-28 16:57:27 +05:30
UpdatePlayerState(playerPawn, PlayerState.HasFinished);
playerPawn.SetPlayerSelectionState(false);
2026-02-11 19:53:16 +05:30
playerGameDatasDict[currentPlayerTypeTurn].totalPawnsFinished++;
2026-02-10 19:24:08 +05:30
playerGameDatasDict[currentPlayerTypeTurn].totalPawnsInFinishingPath--;
2026-02-02 21:20:41 +05:30
Debug.Log($"playerGameDatasDict[currentPlayerTypeTurn].totalPawnsFinished: {playerGameDatasDict[currentPlayerTypeTurn].totalPawnsFinished}");
Debug.Log($"playerGameDatasDict[currentPlayerTypeTurn].playerPawnsDict.Count: {playerGameDatasDict[currentPlayerTypeTurn].playerPawnsDict.Count}");
2026-01-23 12:46:59 +05:30
if (playerGameDatasDict[currentPlayerTypeTurn].totalPawnsFinished == playerGameDatasDict[currentPlayerTypeTurn].playerPawnsDict.Count)
{
CanRollDiceAgain = false;
var playerTypeToRemove = currentPlayerTypeTurn;
2026-02-02 21:20:41 +05:30
allPlayerTypes.ForEach(type => Debug.Log($"before allPlayerTypes: {type}"));
SwitchPlayer();
2026-02-02 21:20:41 +05:30
allPlayerTypes.ForEach(type => Debug.Log($"after allPlayerTypes: {type}"));
if (allPlayerTypes.Contains(playerTypeToRemove))
{
allPlayerTypes.Remove(playerTypeToRemove);
playerDatas.FirstOrDefault(data => data.playerType == playerTypeToRemove).ranking = TotalPlayersInGame - allPlayerTypes.Count;
}
if (allPlayerTypes.Count == 1)
{
// Game is over
var lastUnfinishingPlayerType = allPlayerTypes[0];
2026-01-30 21:51:33 +05:30
allPlayerTypes.RemoveAt(0);
playerDatas.FirstOrDefault(data => data.playerType == lastUnfinishingPlayerType).ranking = TotalPlayersInGame - allPlayerTypes.Count;
// Show Game Over panel
GameManager.OnGameStateChanged(GameState.GameOver);
return;
}
}
else
{
CanRollDiceAgain = true;
2026-01-28 20:57:35 +05:30
if (playerPawn.IsBotPlayer)
CheckDiceRollForBot();
2026-01-30 19:04:14 +05:30
else
{
SetCanRollDiceForUser(true);
2026-02-12 21:34:08 +05:30
UpdateResponseTimerForUser(currentPlayerMaxTime: currentPlayerTurnMaxTime, () => RollDiceForUser());
}
2026-02-12 21:34:08 +05:30
diceRollHandler.DiceView.SetDiceButtonInteraction(CanRollDiceAgain);
}
}
else
{
2026-01-30 19:14:08 +05:30
// activate here
2026-02-02 20:12:00 +05:30
TilesManager.RetrieveFinishingTileBasedOnIndex(currentPlayerTypeTurn, playerPawn.CurrentTileIndex).InitPlayerPawn(playerPawn, currentPlayerTypeTurn);
if (CheckForMaxDiceRollAttempt())
{
return;
}
if (CanRollDiceAgain)
2026-02-12 21:34:08 +05:30
{
SetCanRollDiceForUser(IsUsersTurn());
OnCanRollDiceAgain();
2026-02-12 21:34:08 +05:30
#if UNITY_EDITOR
ShowUpdatedPlayerCountOnTile(playerPawn);
2026-02-12 21:34:08 +05:30
#endif
}
else
SwitchPlayer();
}
2026-01-21 20:27:45 +05:30
}
2026-01-30 19:04:14 +05:30
},
2026-01-21 20:27:45 +05:30
index);
}
2026-01-29 22:16:54 +05:30
private void SetCanRollDiceForUser(bool state)
{
CanRollDiceForUser = state;
2026-01-30 12:54:43 +05:30
Debug.Log($"CAnRollDiceForUser: {CanRollDiceForUser}");
}
2026-01-30 19:04:14 +05:30
2026-02-12 21:34:08 +05:30
#if UNITY_EDITOR
private void ShowUpdatedPlayerCountOnTile(PlayerPawn playerPawn)
2026-01-30 19:04:14 +05:30
{
2026-01-30 23:27:56 +05:30
Tile tile = playerPawn.GetPlayerState() == PlayerState.InFinishingPath ?
2026-02-02 20:12:00 +05:30
TilesManager.RetrieveFinishingTileBasedOnIndex(playerPawn.PlayerType, playerPawn.CurrentTileIndex)
: TilesManager.RetrieveTileBasedOnIndex(playerPawn.CurrentTileIndex);
2026-01-30 23:27:56 +05:30
playerPawn.ShowPlayerCountCanvas(true);
2026-01-30 19:04:14 +05:30
int count = 0;
if (tile.IsSafeZone)
{
count = (tile as SafeTile).GetPlayerPawnsCountInTile(playerPawn.PlayerType);
}
else
{
count = tile.TotalPawnsInTile;
}
Debug.Log($"ShowUpdatedPlayerCountOnTile: {count}");
2026-02-12 21:34:08 +05:30
playerPawn.PlayerIndicatorCanvas.SetPlayerCount(count);
}
private void SetCurrentSelectedPointer()
{
var tempPos = playerBaseHandler.GetPlayerBase(currentPlayerTypeTurn).transform.position;
pointerDebug.position = new Vector3(tempPos.x, 3f, tempPos.z);
2026-01-30 19:04:14 +05:30
}
2026-02-12 21:34:08 +05:30
public void SetDisplayCountForAllAvailPlayers(bool state)
{
if (state)
{
availPlayersToMove.ForEach(pawn => ShowUpdatedPlayerCountOnTile(pawn));
}
else
{
availPlayersToMove.ForEach(pawn => pawn.ShowPlayerCountCanvas(false));
}
}
#endif
2026-02-02 19:27:17 +05:30
public void ResetTileDatasForPlayers()
{
// causes null ref cast exception while player is moving.
// foreach (PlayerData data in playerDatas)
// {
// var pawns = playerGameDatasDict[data.playerType].playerPawnsDict.Values;
//
// foreach (var pawn in pawns)
// {
// if (pawn.GetPlayerState() == PlayerState.InHome) continue;
//
// TilesManager.ResetTileData(pawn.PlayerType, pawn.CurrentTileIndex, pawn.GetPlayerState());
// }
// }
TilesManager.ResetTileDatas();
2026-02-02 19:27:17 +05:30
}
2026-01-30 21:51:33 +05:30
public void ResetData()
{
Debug.Log($"ResetOnSessionEnd()");
diceRollHandler.DiceView.ResetOnSessionEnd();
ResetGameRestartData();
2026-02-12 21:34:08 +05:30
currentPlayerTurnTimer?.KillTimer();
savedCurrentPlayerTurnTimerInst = null;
2026-02-12 21:34:08 +05:30
currentPlayerTurnTimer = null;
onGameResumed = null;
2026-01-30 21:51:33 +05:30
playerDatas = null;
allPlayerTypes = null;
playerGameDatasDict = null;
playerDatas = null;
2026-02-10 19:24:08 +05:30
availPlayersToMove = null;
2026-02-02 17:21:38 +05:30
botTypesInGame = null;
botRuntimeMovementData = null;
2026-01-30 21:51:33 +05:30
}
public void ResetGameRestartData()
{
currentPlayerTurnIndex = 0;
diceSixRollCounter = 0;
}
2026-01-21 20:27:45 +05:30
}