// ---------------------------------------------------------------------------- // // Loadbalancing Framework for Photon - Copyright (C) 2018 Exit Games GmbH // // // Per client in a room, a Player is created. This client's Player is also // known as PhotonClient.LocalPlayer and the only one you might change // properties for. // // developer@photonengine.com // ---------------------------------------------------------------------------- #if UNITY_4_7 || UNITY_5 || UNITY_5_3_OR_NEWER #define SUPPORTED_UNITY #endif namespace Photon.Realtime { using System; using System.Collections; using System.Collections.Generic; using ExitGames.Client.Photon; #if SUPPORTED_UNITY using UnityEngine; #endif #if SUPPORTED_UNITY || NETFX_CORE using Hashtable = ExitGames.Client.Photon.Hashtable; using SupportClass = ExitGames.Client.Photon.SupportClass; #endif /// /// Summarizes a "player" within a room, identified (in that room) by ID (or "actorNumber"). /// /// /// Each player has a actorNumber, valid for that room. It's -1 until assigned by server (and client logic). /// public class Player { /// /// Used internally to identify the masterclient of a room. /// protected internal Room RoomReference { get; set; } /// Backing field for property. private int actorNumber = -1; /// Identifier of this player in current room. Also known as: actorNumber or actorNumber. It's -1 outside of rooms. /// The ID is assigned per room and only valid in that context. It will change even on leave and re-join. IDs are never re-used per room. public int ActorNumber { get { return this.actorNumber; } } /// Only one player is controlled by each client. Others are not local. public readonly bool IsLocal; public bool HasRejoined { get; internal set; } /// Background field for nickName. private string nickName = string.Empty; /// Non-unique nickname of this player. Synced automatically in a room. /// /// A player might change his own playername in a room (it's only a property). /// Setting this value updates the server and other players (using an operation). /// public string NickName { get { return this.nickName; } set { if (!string.IsNullOrEmpty(this.nickName) && this.nickName.Equals(value)) { return; } this.nickName = value; // update a room, if we changed our nickName locally if (this.IsLocal) { this.SetPlayerNameProperty(); } } } /// UserId of the player, available when the room got created with RoomOptions.PublishUserId = true. /// Useful for and blocking slots in a room for expected players (e.g. in ). public string UserId { get; internal set; } /// /// True if this player is the Master Client of the current room. /// public bool IsMasterClient { get { if (this.RoomReference == null) { return false; } return this.ActorNumber == this.RoomReference.MasterClientId; } } /// If this player is active in the room (and getting events which are currently being sent). /// /// Inactive players keep their spot in a room but otherwise behave as if offline (no matter what their actual connection status is). /// The room needs a PlayerTTL != 0. If a player is inactive for longer than PlayerTTL, the server will remove this player from the room. /// For a client "rejoining" a room, is the same as joining it: It gets properties, cached events and then the live events. /// public bool IsInactive { get; protected internal set; } /// Read-only cache for custom properties of player. Set via Player.SetCustomProperties. /// /// Don't modify the content of this Hashtable. Use SetCustomProperties and the /// properties of this class to modify values. When you use those, the client will /// sync values with the server. /// /// public Hashtable CustomProperties { get; set; } /// Can be used to store a reference that's useful to know "by player". /// Example: Set a player's character as Tag by assigning the GameObject on Instantiate. public object TagObject; /// /// Creates a player instance. /// To extend and replace this Player, override LoadBalancingPeer.CreatePlayer(). /// /// NickName of the player (a "well known property"). /// ID or ActorNumber of this player in the current room (a shortcut to identify each player in room) /// If this is the local peer's player (or a remote one). protected internal Player(string nickName, int actorNumber, bool isLocal) : this(nickName, actorNumber, isLocal, null) { } /// /// Creates a player instance. /// To extend and replace this Player, override LoadBalancingPeer.CreatePlayer(). /// /// NickName of the player (a "well known property"). /// ID or ActorNumber of this player in the current room (a shortcut to identify each player in room) /// If this is the local peer's player (or a remote one). /// A Hashtable of custom properties to be synced. Must use String-typed keys and serializable datatypes as values. protected internal Player(string nickName, int actorNumber, bool isLocal, Hashtable playerProperties) { this.IsLocal = isLocal; this.actorNumber = actorNumber; this.NickName = nickName; this.CustomProperties = new Hashtable(); this.InternalCacheProperties(playerProperties); } /// /// Get a Player by ActorNumber (Player.ID). /// /// ActorNumber of the a player in this room. /// Player or null. public Player Get(int id) { if (this.RoomReference == null) { return null; } return this.RoomReference.GetPlayer(id); } /// Gets this Player's next Player, as sorted by ActorNumber (Player.ID). Wraps around. /// Player or null. public Player GetNext() { return GetNextFor(this.ActorNumber); } /// Gets a Player's next Player, as sorted by ActorNumber (Player.ID). Wraps around. /// Useful when you pass something to the next player. For example: passing the turn to the next player. /// The Player for which the next is being needed. /// Player or null. public Player GetNextFor(Player currentPlayer) { if (currentPlayer == null) { return null; } return GetNextFor(currentPlayer.ActorNumber); } /// Gets a Player's next Player, as sorted by ActorNumber (Player.ID). Wraps around. /// Useful when you pass something to the next player. For example: passing the turn to the next player. /// The ActorNumber (Player.ID) for which the next is being needed. /// Player or null. public Player GetNextFor(int currentPlayerId) { if (this.RoomReference == null || this.RoomReference.Players == null || this.RoomReference.Players.Count < 2) { return null; } Dictionary players = this.RoomReference.Players; int nextHigherId = int.MaxValue; // we look for the next higher ID int lowestId = currentPlayerId; // if we are the player with the highest ID, there is no higher and we return to the lowest player's id foreach (int playerid in players.Keys) { if (playerid < lowestId) { lowestId = playerid; // less than any other ID (which must be at least less than this player's id). } else if (playerid > currentPlayerId && playerid < nextHigherId) { nextHigherId = playerid; // more than our ID and less than those found so far. } } //UnityEngine.Debug.LogWarning("Debug. " + currentPlayerId + " lower: " + lowestId + " higher: " + nextHigherId + " "); //UnityEngine.Debug.LogWarning(this.RoomReference.GetPlayer(currentPlayerId)); //UnityEngine.Debug.LogWarning(this.RoomReference.GetPlayer(lowestId)); //if (nextHigherId != int.MaxValue) UnityEngine.Debug.LogWarning(this.RoomReference.GetPlayer(nextHigherId)); return (nextHigherId != int.MaxValue) ? players[nextHigherId] : players[lowestId]; } /// Caches properties for new Players or when updates of remote players are received. Use SetCustomProperties() for a synced update. /// /// This only updates the CustomProperties and doesn't send them to the server. /// Mostly used when creating new remote players, where the server sends their properties. /// protected internal virtual void InternalCacheProperties(Hashtable properties) { if (properties == null || properties.Count == 0 || this.CustomProperties.Equals(properties)) { return; } if (properties.ContainsKey(ActorProperties.PlayerName)) { string nameInServersProperties = (string)properties[ActorProperties.PlayerName]; if (nameInServersProperties != null) { if (this.IsLocal) { // the local playername is different than in the properties coming from the server // so the local nickName was changed and the server is outdated -> update server // update property instead of using the outdated nickName coming from server if (!nameInServersProperties.Equals(this.nickName)) { this.SetPlayerNameProperty(); } } else { this.NickName = nameInServersProperties; } } } if (properties.ContainsKey(ActorProperties.UserId)) { this.UserId = (string)properties[ActorProperties.UserId]; } if (properties.ContainsKey(ActorProperties.IsInactive)) { this.IsInactive = (bool)properties[ActorProperties.IsInactive]; //TURNBASED new well-known propery for players } this.CustomProperties.MergeStringKeys(properties); this.CustomProperties.StripKeysWithNullValues(); } /// /// Brief summary string of the Player: ActorNumber and NickName /// public override string ToString() { return string.Format("#{0:00} '{1}'",this.ActorNumber, this.NickName); } /// /// String summary of the Player: player.ID, name and all custom properties of this user. /// /// /// Use with care and not every frame! /// Converts the customProperties to a String on every single call. /// public string ToStringFull() { return string.Format("#{0:00} '{1}'{2} {3}", this.ActorNumber, this.NickName, this.IsInactive ? " (inactive)" : "", this.CustomProperties.ToStringFull()); } /// /// If players are equal (by GetHasCode, which returns this.ID). /// public override bool Equals(object p) { Player pp = p as Player; return (pp != null && this.GetHashCode() == pp.GetHashCode()); } /// /// Accompanies Equals, using the ID (actorNumber) as HashCode to return. /// public override int GetHashCode() { return this.ActorNumber; } /// /// Used internally, to update this client's playerID when assigned (doesn't change after assignment). /// protected internal void ChangeLocalID(int newID) { if (!this.IsLocal) { //Debug.LogError("ERROR You should never change Player IDs!"); return; } this.actorNumber = newID; } /// /// Updates and synchronizes this Player's Custom Properties. Optionally, expectedProperties can be provided as condition. /// /// /// Custom Properties are a set of string keys and arbitrary values which is synchronized /// for the players in a Room. They are available when the client enters the room, as /// they are in the response of OpJoin and OpCreate. /// /// Custom Properties either relate to the (current) Room or a Player (in that Room). /// /// Both classes locally cache the current key/values and make them available as /// property: CustomProperties. This is provided only to read them. /// You must use the method SetCustomProperties to set/modify them. /// /// Any client can set any Custom Properties anytime (when in a room). /// It's up to the game logic to organize how they are best used. /// /// You should call SetCustomProperties only with key/values that are new or changed. This reduces /// traffic and performance. /// /// Unless you define some expectedProperties, setting key/values is always permitted. /// In this case, the property-setting client will not receive the new values from the server but /// instead update its local cache in SetCustomProperties. /// /// If you define expectedProperties, the server will skip updates if the server property-cache /// does not contain all expectedProperties with the same values. /// In this case, the property-setting client will get an update from the server and update it's /// cached key/values at about the same time as everyone else. /// /// The benefit of using expectedProperties can be only one client successfully sets a key from /// one known value to another. /// As example: Store who owns an item in a Custom Property "ownedBy". It's 0 initally. /// When multiple players reach the item, they all attempt to change "ownedBy" from 0 to their /// actorNumber. If you use expectedProperties {"ownedBy", 0} as condition, the first player to /// take the item will have it (and the others fail to set the ownership). /// /// Properties get saved with the game state for Turnbased games (which use IsPersistent = true). /// /// Hashtable of Custom Properties to be set. /// If non-null, these are the property-values the server will check as condition for this update. /// Defines if this SetCustomProperties-operation gets forwarded to your WebHooks. Client must be in room. /// /// False if propertiesToSet is null or empty or have zero string keys. /// True in offline mode even if expectedProperties or webFlags are used. /// If not in a room, returns true if local player and expectedValues and webFlags are null. /// (Use this to cache properties to be sent when joining a room). /// Otherwise, returns if this operation could be sent to the server. /// public bool SetCustomProperties(Hashtable propertiesToSet, Hashtable expectedValues = null, WebFlags webFlags = null) { if (propertiesToSet == null || propertiesToSet.Count == 0) { return false; } Hashtable customProps = propertiesToSet.StripToStringKeys() as Hashtable; if (this.RoomReference != null) { if (this.RoomReference.IsOffline) { if (customProps.Count == 0) { return false; } this.CustomProperties.Merge(customProps); this.CustomProperties.StripKeysWithNullValues(); // invoking callbacks this.RoomReference.LoadBalancingClient.InRoomCallbackTargets.OnPlayerPropertiesUpdate(this, customProps); return true; } else { Hashtable customPropsToCheck = expectedValues.StripToStringKeys() as Hashtable; // send (sync) these new values if in online room return this.RoomReference.LoadBalancingClient.OpSetPropertiesOfActor(this.actorNumber, customProps, customPropsToCheck, webFlags); } } if (this.IsLocal) { if (customProps.Count == 0) { return false; } if (expectedValues == null && webFlags == null) { this.CustomProperties.Merge(customProps); this.CustomProperties.StripKeysWithNullValues(); return true; } } return false; } /// Uses OpSetPropertiesOfActor to sync this player's NickName (server is being updated with this.NickName). private bool SetPlayerNameProperty() { if (this.RoomReference != null && !this.RoomReference.IsOffline) { Hashtable properties = new Hashtable(); properties[ActorProperties.PlayerName] = this.nickName; return this.RoomReference.LoadBalancingClient.OpSetPropertiesOfActor(this.ActorNumber, properties); } return false; } } }