diff --git a/DataCollection/gamedata.json b/DataCollection/gamedata.json index 1cd3777..2c7f3ab 100644 --- a/DataCollection/gamedata.json +++ b/DataCollection/gamedata.json @@ -35,6 +35,13 @@ "location_format":" You are%META% ", "tile_format":"%TILENAME%", "nothing_message":"^LYou see nothing on the ground of interest.^R1", + "nearby":{ + "players_nearby":"Players Nearby:", + "east":"East:", + "west":"West:", + "south":"South:", + "north":"North:" + }, "seperator":"
", "areas":{ "bridge_text":"A bridge crosses over the water to another isle. That'll save you some travel costs!" @@ -158,7 +165,7 @@ "near_format":"%USERNAME%: %MESSAGE%", "isle_format":"%USERNAME%: %MESSAGE%", "dm_format":"%USERNAME%: %MESSAGE%", - "dm_format_moderator":"%USERNAME%[mod: %MESSAGE%" + "dm_format_moderator":"%USERNAME%[mod]: %MESSAGE%" }, "for_sender":{ @@ -237,7 +244,8 @@ "violation_points_required":10, "violation_format":"Your last chat was flagged as a possible violation of chat rules to be reviewed by an admin. If it is severe or you have many violations your account may be suspended. We require at least %AMOUNT% violation points before examining an account, so if it was a simple misspelling do not worry. Remember this is a family friendly game! Thanks for keeping it as such! (REASON: %REASON%)", "password_included":"Your Password was included in the chat you attempted to send! DO NOT Give ANYONE your password! If someone was asking for your password, please report them immediately!", - "caps_notice":"Please do not use all CAPS, it looks as if you are yelling." + "caps_notice":"Please do not use all CAPS, it looks as if you are yelling.", + "pm_sound":"PM" } }, "mod_splatterball_game":{ diff --git a/DataCollection/notes.txt b/DataCollection/notes.txt index ad74d81..2380d01 100644 --- a/DataCollection/notes.txt +++ b/DataCollection/notes.txt @@ -33,4 +33,5 @@ Not implemented: admin_format FOR SENDER is unconfirmed. +Players Nearby:
North: GinoIsLove
Near radius is a square, 15x19 \ No newline at end of file diff --git a/Horse Isle Server/Horse Isle Server/Chat.cs b/Horse Isle Server/Horse Isle Server/Chat.cs index 2439a52..0c1cb92 100644 --- a/Horse Isle Server/Horse Isle Server/Chat.cs +++ b/Horse Isle Server/Horse Isle Server/Chat.cs @@ -31,7 +31,8 @@ namespace Horse_Isle_Server Ads = 0x1D, Near = 0x15, Buddies = 0x17, - Isle = 0x24, + Here = 0x18, + Isle = 0x1A, Dm = 0x16, Mod = 0x1c, Admin = 0x1b @@ -95,11 +96,12 @@ namespace Horse_Isle_Server case ChatChannel.Isle: return PacketBuilder.CHAT_BOTTOM_LEFT; case ChatChannel.Buddies: + case ChatChannel.Here: case ChatChannel.Admin: case ChatChannel.Mod: return PacketBuilder.CHAT_BOTTOM_RIGHT; case ChatChannel.Dm: - return PacketBuilder.CHAT_BTMR_W_DM_SFX; + return PacketBuilder.CHAT_DM_RIGHT; default: Logger.ErrorPrint("unknown channel: " + (byte)channel); return PacketBuilder.CHAT_BOTTOM_LEFT; @@ -112,7 +114,7 @@ namespace Horse_Isle_Server if(message.Contains('|')) { string recipiantName = message.Split('|')[0]; - return recipiantName.Substring(2); + return recipiantName; } else { @@ -120,6 +122,19 @@ namespace Horse_Isle_Server } } + public static string GetDmMessage(string message) + { + if (message.Contains('|')) + { + string messageStr = message.Split('|')[1]; + return messageStr; + } + else + { + return message; + } + } + public static Client[] GetRecipiants(User user, ChatChannel channel, string to=null) { if (channel == ChatChannel.All) @@ -162,6 +177,50 @@ namespace Horse_Isle_Server return recipiants.ToArray(); } + if (channel == ChatChannel.Isle) + { + List recipiants = new List(); + if(World.InIsle(user.X,user.Y)) + { + User[] usersInSile = Server.GetUsersUsersInIsle(World.GetIsle(user.X, user.Y), true, false); + foreach (User userInIsle in usersInSile) + { + if (user.Id != userInIsle.Id) + recipiants.Add(userInIsle.LoggedinClient); + } + return recipiants.ToArray(); + } + else + { + return new Client[0]; + } + + } + + if (channel == ChatChannel.Here) + { + List recipiants = new List(); + User[] usersHere = Server.GetUsersAt(user.X, user.Y, true, false); + foreach (User userHere in usersHere) + { + if (user.Id != userHere.Id) + recipiants.Add(userHere.LoggedinClient); + } + return recipiants.ToArray(); + } + + if (channel == ChatChannel.Near) + { + List recipiants = new List(); + User[] nearbyUsers = Server.GetNearbyUsers(user.X, user.Y, true, false); + foreach (User nearbyUser in nearbyUsers) + { + if (user.Id != nearbyUser.Id) + recipiants.Add(nearbyUser.LoggedinClient); + } + return recipiants.ToArray(); + } + if (channel == ChatChannel.Mod) { if (!user.Moderator || !user.Administrator) // No mod chat for non-mods! @@ -210,7 +269,7 @@ namespace Horse_Isle_Server { if (client.LoggedIn) if (!client.LoggedinUser.MutePrivateMessage) - if (client.LoggedinUser.Username != to) + if (client.LoggedinUser.Username == to) recipiants.Add(client); } return recipiants.ToArray(); @@ -261,22 +320,28 @@ namespace Horse_Isle_Server return Messages.FormatDirectMessageForMod(user.Username, message); else return Messages.FormatDirectMessage(user.Username, message); + case ChatChannel.Near: + return Messages.FormatNearbyChatMessage(user.Username, message); + case ChatChannel.Isle: + return Messages.FormatIsleChatMessage(user.Username, message); + case ChatChannel.Here: + return Messages.FormatHereChatMessage(user.Username, message); case ChatChannel.Mod: if (user.Moderator || user.Administrator) return Messages.FormatModChatMessage(user.Username, message); else - return "Hacker!"; + return user.Username+" is a hacker! (Sent in mod channel without being a mod) Maybe ban?"; case ChatChannel.Admin: if (user.Administrator) return Messages.FormatAdminChatMessage(user.Username, message); else - return "Hacker!"; + return user.Username + " is a hacker! (Sent in admin channel without being a admin) Maybe ban?"; default: Logger.ErrorPrint(user.Username + " is trying to end a message in unknown channel " + channel.ToString("X")); return "not implemented yet :("; } } - public static string FormatChatForSender(User user, ChatChannel channel, string message) + public static string FormatChatForSender(User user, ChatChannel channel, string message, string dmRecipiant=null) { switch (channel) { @@ -289,10 +354,25 @@ namespace Horse_Isle_Server return Messages.FormatAdsChatMessage(user.Username, message); case ChatChannel.Buddies: return Messages.FormatBuddyChatMessageForSender(user.Friends.Count, user.Username, message); + case ChatChannel.Isle: + int inIsle = 0; + if (World.InIsle(user.X, user.Y)) + inIsle = Server.GetUsersUsersInIsle(World.GetIsle(user.X, user.Y), false, false).Length -1; + return Messages.FormatIsleChatMessageForSender(inIsle, user.Username, message); + case ChatChannel.Here: + int usersHere = Server.GetUsersAt(user.X, user.Y, false, false).Length -1; + return Messages.FormatHereChatMessageForSender(usersHere, user.Username, message); + case ChatChannel.Near: + int nearbyUsers = Server.GetNearbyUsers(user.X, user.Y, false, false).Length -1; + return Messages.FormatNearChatMessageForSender(nearbyUsers, user.Username, message); case ChatChannel.Mod: - return Messages.FormatModChatForSender(Server.GetNumberOfModsOnline(), user.Username, message); + int modsOnline = Server.GetNumberOfModsOnline(); + return Messages.FormatModChatForSender(modsOnline, user.Username, message); case ChatChannel.Admin: - return Messages.FormatAdminChatForSender(Server.GetNumberOfAdminsOnline(),user.Username, message); + int adminsOnline = Server.GetNumberOfAdminsOnline(); + return Messages.FormatAdminChatForSender(adminsOnline, user.Username, message); + case ChatChannel.Dm: + return Messages.FormatDirectChatMessageForSender(user.Username, dmRecipiant, message); default: Logger.ErrorPrint(user.Username + " is trying to end a message in unknown channel " + channel.ToString("X")); return "not implemented yet :("; @@ -304,22 +384,27 @@ namespace Horse_Isle_Server // Check if contains password. if (message.ToLower().Contains(user.Password.ToLower())) - { return Messages.PasswordNotice; - } - - + // Check if ALL CAPS - if (message.Contains(' ')) // hi1 apparently doesnt care about caps if its all 1 word? + string[] wordsSaid; + if (message.Contains(' ')) + wordsSaid = message.Split(' '); + else + wordsSaid = new string[] { message }; + + foreach (string word in wordsSaid) { - string[] wordsSaid = message.Split(' '); - foreach (string word in wordsSaid) + string lettersOnly = ""; + foreach(char c in word) { - if (word.ToUpper() == word) + if((byte)c >= (byte)'A' && (byte)c <= (byte)'z') // is letter { - return Messages.CapsNotice; + lettersOnly += c; } } + if (lettersOnly.ToUpper() == lettersOnly && lettersOnly.Length >= 5) + return Messages.CapsNotice; } return null; diff --git a/Horse Isle Server/Horse Isle Server/Client.cs b/Horse Isle Server/Horse Isle Server/Client.cs index ba1adb6..da4a46c 100644 --- a/Horse Isle Server/Horse Isle Server/Client.cs +++ b/Horse Isle Server/Horse Isle Server/Client.cs @@ -55,7 +55,7 @@ namespace Horse_Isle_Server } public void Login(int id) { - LoggedinUser = new User(id); + LoggedinUser = new User(this,id); LoggedIn = true; updateTimer = new Timer(new TimerCallback(updateTimerTick), null, updateInterval, updateInterval); diff --git a/Horse Isle Server/Horse Isle Server/Gamedata.cs b/Horse Isle Server/Horse Isle Server/Gamedata.cs index 6447dc7..f301fcb 100644 --- a/Horse Isle Server/Horse Isle Server/Gamedata.cs +++ b/Horse Isle Server/Horse Isle Server/Gamedata.cs @@ -131,6 +131,10 @@ namespace Horse_Isle_Server Messages.GlobalChatFormatForModerators = gameData.messages.chat.for_others.global_format_moderator; Messages.DirectChatFormatForModerators = gameData.messages.chat.for_others.dm_format_moderator; + + Messages.HereChatFormat = gameData.messages.chat.for_others.here_format; + Messages.IsleChatFormat = gameData.messages.chat.for_others.isle_format; + Messages.NearChatFormat = gameData.messages.chat.for_others.near_format; Messages.GlobalChatFormat = gameData.messages.chat.for_others.global_format; Messages.AdsChatFormat = gameData.messages.chat.for_others.ads_format; Messages.DirectChatFormat = gameData.messages.chat.for_others.dm_format; @@ -159,6 +163,12 @@ namespace Horse_Isle_Server Messages.TileFormat = gameData.messages.meta.tile_format; Messages.NothingMessage = gameData.messages.meta.nothing_message; + Messages.NearbyPlayers = gameData.messages.meta.nearby.players_nearby; + Messages.North = gameData.messages.meta.nearby.north; + Messages.East = gameData.messages.meta.nearby.east; + Messages.South = gameData.messages.meta.nearby.south; + Messages.West = gameData.messages.meta.nearby.west; + // Map Data Map.OverlayTileDepth = gameData.tile_paramaters.overlay_tiles.tile_depth.ToObject(); @@ -171,6 +181,7 @@ namespace Horse_Isle_Server Messages.IdleKickMessageFormat = gameData.messages.disconnect.client_timeout.kick_message; Messages.IdleWarningFormat = gameData.messages.disconnect.client_timeout.warn_message; + Chat.PrivateMessageSound = gameData.messages.chat.pm_sound; Server.IdleWarning = gameData.messages.disconnect.client_timeout.warn_after; Server.IdleTimeout = gameData.messages.disconnect.client_timeout.kick_after; diff --git a/Horse Isle Server/Horse Isle Server/Messages.cs b/Horse Isle Server/Horse Isle Server/Messages.cs index b71fefa..21dbf32 100644 --- a/Horse Isle Server/Horse Isle Server/Messages.cs +++ b/Horse Isle Server/Horse Isle Server/Messages.cs @@ -23,6 +23,9 @@ namespace Horse_Isle_Server public static string GlobalChatFormat; public static string AdsChatFormat; public static string BuddyChatFormat; + public static string NearChatFormat; + public static string IsleChatFormat; + public static string HereChatFormat; public static string DirectChatFormat; public static string ModChatFormat; public static string AdminChatFormat; @@ -30,9 +33,9 @@ namespace Horse_Isle_Server public static string GlobalChatFormatForModerators; public static string DirectChatFormatForModerators; - public static string HereChatFormatForSender; public static string IsleChatFormatForSender; public static string NearChatFormatForSender; + public static string HereChatFormatForSender; public static string BuddyChatFormatForSender; public static string DirectChatFormatForSender; public static string AdminChatFormatForSender; @@ -49,6 +52,12 @@ namespace Horse_Isle_Server public static string AreaFormat; public static string LocationFormat; + public static string NearbyPlayers; + public static string North; + public static string East; + public static string South; + public static string West; + public static string TileFormat; public static string NothingMessage; public static string Seperator; @@ -73,6 +82,21 @@ namespace Horse_Isle_Server return BuddyChatFormat.Replace("%USERNAME%", username).Replace("%MESSAGE%", message); } + public static string FormatIsleChatMessage(string username, string message) + { + return IsleChatFormat.Replace("%USERNAME%", username).Replace("%MESSAGE%", message); + } + + public static string FormatNearbyChatMessage(string username, string message) + { + return NearChatFormat.Replace("%USERNAME%", username).Replace("%MESSAGE%", message); + } + + public static string FormatHereChatMessage(string username, string message) + { + return HereChatFormat.Replace("%USERNAME%", username).Replace("%MESSAGE%", message); + } + public static string FormatDirectMessage(string username, string message) { return DirectChatFormat.Replace("%USERNAME%", username).Replace("%MESSAGE%", message); @@ -108,6 +132,18 @@ namespace Horse_Isle_Server { return BuddyChatFormatForSender.Replace("%USERNAME%", username).Replace("%MESSAGE%", message).Replace("%AMOUNT%", numbBuddies.ToString()); } + public static string FormatHereChatMessageForSender(int numbHere, string username, string message) + { + return HereChatFormatForSender.Replace("%USERNAME%", username).Replace("%MESSAGE%", message).Replace("%AMOUNT%", numbHere.ToString()); + } + public static string FormatNearChatMessageForSender(int numbNear, string username, string message) + { + return NearChatFormatForSender.Replace("%USERNAME%", username).Replace("%MESSAGE%", message).Replace("%AMOUNT%", numbNear.ToString()); + } + public static string FormatIsleChatMessageForSender(int numbIsle, string username, string message) + { + return IsleChatFormatForSender.Replace("%USERNAME%", username).Replace("%MESSAGE%", message).Replace("%AMOUNT%", numbIsle.ToString()); + } public static string FormatAdminChatForSender(int numbAdmins, string username, string message) { @@ -152,25 +188,87 @@ namespace Horse_Isle_Server return IdleKickMessageFormat.Replace("%KICK%", Server.IdleTimeout.ToString()); } // Meta - public static string FormatLocationData(int x, int y) + + private static string buildLocationString(int x, int y) { string locationString = ""; - string message = ""; - if(World.InArea(x,y)) + + if (World.InArea(x, y)) locationString += AreaFormat.Replace("%AREA%", World.GetArea(x, y).Name); if (World.InTown(x, y)) locationString += TownFormat.Replace("%TOWN%", World.GetTown(x, y).Name); if (World.InIsle(x, y)) locationString += IsleFormat.Replace("%ISLE%", World.GetIsle(x, y).Name); + if (locationString != "") + locationString = LocationFormat.Replace("%META%", locationString); + return locationString; + } - if(locationString != "") - message += LocationFormat.Replace("%META%", locationString); + private static string buildNearbyString(int x, int y) + { + string playersNearby = ""; + User[] nearbyUsers = Server.GetNearbyUsers(x, y, true, true); + if(nearbyUsers.Length > 1) + { + playersNearby += NearbyPlayers; + playersNearby += Seperator; + + string usersWest = ""; + string usersNorth = ""; + string usersEast = ""; + string usersSouth = ""; + foreach (User nearbyUser in nearbyUsers) + { + if (nearbyUser.X < x) + { + usersWest += " " + nearbyUser.Username + " "; + } + else if(nearbyUser.X > x) + { + usersEast += " " + nearbyUser.Username + " "; + } + else if (nearbyUser.Y > y) + { + usersSouth += " " + nearbyUser.Username + " "; + } + else if (nearbyUser.Y < y) + { + usersNorth += " " + nearbyUser.Username + " "; + } + } + + if(usersEast != "") + playersNearby += " " + East + usersEast + Seperator; + if (usersWest != "") + playersNearby += " " + West + usersWest + Seperator; + if (usersSouth != "") + playersNearby += " " + South + usersSouth + Seperator; + if (usersNorth != "") + playersNearby += " " + North + usersNorth + Seperator; + + + + } + + return playersNearby; + + } + public static string BuildMetaInfo(int x, int y) + { + // You are in + string message = buildLocationString(x, y); + + // Nearby + message += Seperator + buildNearbyString(x, y); + + // Dropped Items int[] itemIds = World.GetDroppedItems(x, y); if (itemIds.Length == 0) message += NothingMessage; return message; } + } } diff --git a/Horse Isle Server/Horse Isle Server/PacketBuilder.cs b/Horse Isle Server/Horse Isle Server/PacketBuilder.cs index bfea041..1358d8a 100644 --- a/Horse Isle Server/Horse Isle Server/PacketBuilder.cs +++ b/Horse Isle Server/Horse Isle Server/PacketBuilder.cs @@ -21,6 +21,7 @@ namespace Horse_Isle_Server public const byte PACKET_AREA_DEFS = 0x79; public const byte PACKET_ANNOUNCEMENT = 0x7E; public const byte PACKET_TILE_FLAGS = 0x75; + public const byte PACKET_PLAYSOUND = 0x23; public const byte PACKET_KEEP_ALIVE = 0x7C; public const byte PACKET_PROFILE = 0x18; public const byte PACKET_KICK = 0x80; @@ -47,7 +48,7 @@ namespace Horse_Isle_Server public const byte CHAT_BOTTOM_LEFT = 0x14; public const byte CHAT_BOTTOM_RIGHT = 0x15; - public const byte CHAT_BTMR_W_DM_SFX = 0x16; + public const byte CHAT_DM_RIGHT = 0x16; public const byte LOGIN_INVALID_USER_PASS = 0x15; public const byte LOGIN_SUCCESS = 0x14; @@ -60,6 +61,22 @@ namespace Horse_Isle_Server public const byte DIRECTION_NONE = 10; + public static byte[] CreatePlaysoundPacket(string sound) + { + MemoryStream ms = new MemoryStream(); + ms.WriteByte(PACKET_PLAYSOUND); + + byte[] strBytes = Encoding.UTF8.GetBytes(sound); + ms.Write(strBytes, 0x00, strBytes.Length); + + ms.WriteByte(PACKET_TERMINATOR); + + ms.Seek(0x00, SeekOrigin.Begin); + byte[] Packet = ms.ToArray(); + ms.Dispose(); + + return Packet; + } public static byte[] CreatePlayerLeavePacket(string username) { MemoryStream ms = new MemoryStream(); @@ -573,7 +590,7 @@ namespace Horse_Isle_Server public static byte[] CreateAreaMessage(int x, int y) { - string locationStr = Messages.FormatLocationData(x, y); + string locationStr = Messages.BuildMetaInfo(x, y); return CreatePlaceInfo(locationStr); } public static byte[] CreateMotd() diff --git a/Horse Isle Server/Horse Isle Server/Server.cs b/Horse Isle Server/Horse Isle Server/Server.cs index 661096d..e48a15f 100644 --- a/Horse Isle Server/Horse Isle Server/Server.cs +++ b/Horse Isle Server/Horse Isle Server/Server.cs @@ -17,16 +17,16 @@ namespace Horse_Isle_Server public static Client[] ConnectedClients // Done to prevent Enumerator Changed errors. - { + { get { - return connectedClients.ToArray(); - } + return connectedClients.ToArray(); + } } public static int IdleTimeout; public static int IdleWarning; - + // used for world time, private static int gameTickSpeed = 4320; // Changing this to ANYTHING else will cause desync with the client. @@ -71,11 +71,11 @@ namespace Horse_Isle_Server byte[] AreaMessage = PacketBuilder.CreateAreaMessage(user.X, user.Y); sender.SendPacket(AreaMessage); - foreach(Client client in ConnectedClients) + foreach (Client client in ConnectedClients) { if (client.LoggedIn) { - if(client.LoggedinUser.Id != user.Id) + if (client.LoggedinUser.Id != user.Id) { byte[] PlayerInfo = PacketBuilder.CreatePlayerInfoUpdateOrCreate(client.LoggedinUser.X, client.LoggedinUser.Y, client.LoggedinUser.Facing, client.LoggedinUser.CharacterId, client.LoggedinUser.Username); sender.SendPacket(PlayerInfo); @@ -83,6 +83,10 @@ namespace Horse_Isle_Server } } + foreach (User nearbyUser in Server.GetNearbyUsers(sender.LoggedinUser.X, sender.LoggedinUser.Y, false, false)) + if (nearbyUser.Id != sender.LoggedinUser.Id) + UpdateArea(nearbyUser.LoggedinClient); + byte[] IsleData = PacketBuilder.CreatePlaceData(World.Isles.ToArray(), World.Towns.ToArray(), World.Areas.ToArray()); sender.SendPacket(IsleData); @@ -94,7 +98,7 @@ namespace Horse_Isle_Server } - + public static void OnKeepAlive(Client sender, byte[] packet) { if (!sender.LoggedIn) @@ -108,9 +112,9 @@ namespace Horse_Isle_Server return; } - if(packet[1] == PacketBuilder.PACKET_CLIENT_TERMINATOR) + if (packet[1] == PacketBuilder.PACKET_CLIENT_TERMINATOR) { - Logger.DebugPrint("Sending "+ sender.LoggedinUser.Username +" updated info..."); + Logger.DebugPrint("Sending " + sender.LoggedinUser.Username + " updated info..."); UpdatePlayer(sender); } } @@ -121,19 +125,19 @@ namespace Horse_Isle_Server Logger.ErrorPrint(sender.RemoteIp + " Requested to change profile page when not logged in."); return; } - if(packet.Length < 2) + if (packet.Length < 2) { Logger.ErrorPrint(sender.LoggedinUser.Username + " Sent an invalid Profile Packet"); return; } - + byte method = packet[1]; - if(method == PacketBuilder.VIEW_PROFILE) + if (method == PacketBuilder.VIEW_PROFILE) { byte[] profilePacket = PacketBuilder.CreateProfilePacket(sender.LoggedinUser.ProfilePage); sender.SendPacket(profilePacket); } - else if(method == PacketBuilder.SAVE_PROFILE) + else if (method == PacketBuilder.SAVE_PROFILE) { string packetStr = Encoding.UTF8.GetString(packet); @@ -173,7 +177,7 @@ namespace Horse_Isle_Server User loggedInUser = sender.LoggedinUser; byte movementDirection = packet[1]; - if(movementDirection == PacketBuilder.MOVE_UP) + if (movementDirection == PacketBuilder.MOVE_UP) { loggedInUser.Facing = PacketBuilder.DIRECTION_UP; if (Map.CheckPassable(loggedInUser.X, loggedInUser.Y - 1)) @@ -189,7 +193,7 @@ namespace Horse_Isle_Server sender.SendPacket(moveUpResponse); } } - else if(movementDirection == PacketBuilder.MOVE_LEFT) + else if (movementDirection == PacketBuilder.MOVE_LEFT) { loggedInUser.Facing = PacketBuilder.DIRECTION_LEFT; if (Map.CheckPassable(loggedInUser.X - 1, loggedInUser.Y)) @@ -204,7 +208,7 @@ namespace Horse_Isle_Server sender.SendPacket(moveLeftResponse); } } - else if(movementDirection == PacketBuilder.MOVE_RIGHT) + else if (movementDirection == PacketBuilder.MOVE_RIGHT) { loggedInUser.Facing = PacketBuilder.DIRECTION_RIGHT; if (Map.CheckPassable(loggedInUser.X + 1, loggedInUser.Y)) @@ -219,7 +223,7 @@ namespace Horse_Isle_Server sender.SendPacket(moveLeftResponse); } } - else if(movementDirection == PacketBuilder.MOVE_DOWN) + else if (movementDirection == PacketBuilder.MOVE_DOWN) { loggedInUser.Facing = PacketBuilder.DIRECTION_DOWN; if (Map.CheckPassable(loggedInUser.X, loggedInUser.Y + 1)) @@ -239,8 +243,11 @@ namespace Horse_Isle_Server UpdateUserInfo(sender.LoggedinUser); UpdateArea(sender); + foreach (User nearbyUser in Server.GetNearbyUsers(sender.LoggedinUser.X, sender.LoggedinUser.Y, false, false)) + if(nearbyUser.Id != sender.LoggedinUser.Id) + UpdateArea(nearbyUser.LoggedinClient); } - + public static void OnChatPacket(Client sender, byte[] packet) { if (!sender.LoggedIn) @@ -249,7 +256,7 @@ namespace Horse_Isle_Server return; } - if(packet.Length < 4) + if (packet.Length < 4) { Logger.ErrorPrint(sender.RemoteIp + " Sent an invalid chat packet."); return; @@ -257,19 +264,25 @@ namespace Horse_Isle_Server string packetStr = Encoding.UTF8.GetString(packet); - + Chat.ChatChannel channel = (Chat.ChatChannel)packet[1]; string message = packetStr.Substring(2, packetStr.Length - 4); + Logger.DebugPrint(sender.LoggedinUser.Username + " Attempting to say '" + message + "' in channel: " + channel.ToString()); string nameTo = null; if (channel == Chat.ChatChannel.Dm) + { nameTo = Chat.GetDmRecipiant(message); + message = Chat.GetDmMessage(message); + } + if (message == "") + return; Object violationReason = Chat.FilterMessage(message); - if (violationReason != null) + if (violationReason != null) { sender.LoggedinUser.ChatViolations += 1; string chatViolationMessage = Messages.FormatGlobalChatViolationMessage((Chat.Reason)violationReason); @@ -282,7 +295,7 @@ namespace Horse_Isle_Server message = Chat.DoCorrections(message); message = Chat.EscapeMessage(message); - + string failedReason = Chat.NonViolationChecks(sender.LoggedinUser, message); if (failedReason != null) { @@ -294,20 +307,23 @@ namespace Horse_Isle_Server Client[] recipiants = Chat.GetRecipiants(sender.LoggedinUser, channel, nameTo); // Finally send chat message. - string formattedMessage = Chat.FormatChatForOthers(sender.LoggedinUser,channel,message); - string formattedMessageSender = Chat.FormatChatForSender(sender.LoggedinUser, channel, message); + string formattedMessage = Chat.FormatChatForOthers(sender.LoggedinUser, channel, message); + string formattedMessageSender = Chat.FormatChatForSender(sender.LoggedinUser, channel, message, nameTo); byte[] chatPacketOthers = PacketBuilder.CreateChat(formattedMessage, chatSide); byte[] chatPacketSender = PacketBuilder.CreateChat(formattedMessageSender, chatSide); + byte[] playDmSound = PacketBuilder.CreatePlaysoundPacket(Chat.PrivateMessageSound); // Send to clients ... foreach (Client recipiant in recipiants) { recipiant.SendPacket(chatPacketOthers); + if (channel == Chat.ChatChannel.Dm) + recipiant.SendPacket(playDmSound); } // Send to sender sender.SendPacket(chatPacketSender); } - + public static void OnLoginRequest(Client sender, byte[] packet) { Logger.DebugPrint("Login request received from: " + sender.RemoteIp); @@ -320,7 +336,7 @@ namespace Horse_Isle_Server return; } - if(packet[1] != PacketBuilder.PACKET_CLIENT_TERMINATOR) + if (packet[1] != PacketBuilder.PACKET_CLIENT_TERMINATOR) { string[] loginParts = loginRequestString.Split('|'); if (loginParts.Length < 3) @@ -345,14 +361,14 @@ namespace Horse_Isle_Server byte[] ResponsePacket = PacketBuilder.CreateLoginPacket(true); sender.SendPacket(ResponsePacket); - Logger.DebugPrint(sender.RemoteIp + " Logged into : " + sender.LoggedinUser.Username + " (ADMIN: " + sender.LoggedinUser.Administrator + " MOD: " + sender.LoggedinUser.Moderator+")"); + Logger.DebugPrint(sender.RemoteIp + " Logged into : " + sender.LoggedinUser.Username + " (ADMIN: " + sender.LoggedinUser.Administrator + " MOD: " + sender.LoggedinUser.Moderator + ")"); // Send login message byte[] loginMessageBytes = PacketBuilder.CreateChat(Messages.FormatLoginMessage(sender.LoggedinUser.Username), PacketBuilder.CHAT_BOTTOM_LEFT); foreach (Client client in ConnectedClients) if (client.LoggedIn) if (!client.LoggedinUser.MuteLogins) - if(client.LoggedinUser.Id != userId) + if (client.LoggedinUser.Id != userId) client.SendPacket(loginMessageBytes); UpdateUserInfo(sender.LoggedinUser); @@ -370,7 +386,7 @@ namespace Horse_Isle_Server public static void OnDisconnect(Client sender) { - if(sender.LoggedIn) + if (sender.LoggedIn) { // Send disconnect message byte[] logoutMessageBytes = PacketBuilder.CreateChat(Messages.FormatLogoutMessage(sender.LoggedinUser.Username), PacketBuilder.CHAT_BOTTOM_LEFT); @@ -387,7 +403,7 @@ namespace Horse_Isle_Server client.SendPacket(playerRemovePacket); } - connectedClients.Remove(sender); + connectedClients.Remove(sender); } public static void UpdateArea(Client forClient) @@ -429,21 +445,91 @@ namespace Horse_Isle_Server public static void UpdateUserInfo(User user) { byte[] playerInfoBytes = PacketBuilder.CreatePlayerInfoUpdateOrCreate(user.X, user.Y, user.Facing, user.CharacterId, user.Username); + + + + List users = new List(); foreach (Client client in ConnectedClients) if (client.LoggedIn) + { if (client.LoggedinUser.Id != user.Id) client.SendPacket(playerInfoBytes); + } + + } - public static int GetNumberOfPlayers() + + public static User[] GetUsersUsersInIsle(World.Isle isle, bool includeStealth = false, bool includeMuted = false) + { + List usersInIsle = new List(); + foreach (Client client in ConnectedClients) + if (client.LoggedIn) + { + if (!includeStealth && client.LoggedinUser.Stealth) + continue; + if (!includeMuted && client.LoggedinUser.MuteIsland) + continue; + if (World.InIsle(client.LoggedinUser.X, client.LoggedinUser.Y)) + if (World.GetIsle(client.LoggedinUser.X, client.LoggedinUser.Y).Name == isle.Name) + usersInIsle.Add(client.LoggedinUser); + } + + return usersInIsle.ToArray(); + } + + public static User[] GetUsersAt(int x, int y, bool includeStealth = false, bool includeMuted = false) + { + List usersHere = new List(); + foreach(Client client in ConnectedClients) + { + if(client.LoggedIn) + { + if (!includeStealth && client.LoggedinUser.Stealth) + continue; + if (!includeMuted && client.LoggedinUser.MuteNear) + continue; + if (client.LoggedinUser.X == x && client.LoggedinUser.Y == y) + usersHere.Add(client.LoggedinUser); + } + } + return usersHere.ToArray(); + } + + public static User[] GetNearbyUsers(int x, int y, bool includeStealth=false, bool includeMuted=false) + { + int startX = x - 15; + int endX = x + 15; + int startY = y - 19; + int endY = y + 19; + List usersNearby = new List(); + + foreach (Client client in ConnectedClients) + if (client.LoggedIn) + { + if (!includeStealth && client.LoggedinUser.Stealth) + continue; + if (!includeMuted && client.LoggedinUser.MuteNear) + continue; + if (startX <= client.LoggedinUser.X && endX >= client.LoggedinUser.X && startY <= client.LoggedinUser.Y && endY >= client.LoggedinUser.Y) + usersNearby.Add(client.LoggedinUser); + } + + return usersNearby.ToArray(); + } + + public static int GetNumberOfPlayers(bool includeStealth=false) { int count = 0; foreach(Client client in ConnectedClients) - { if (client.LoggedIn) - if(!client.LoggedinUser.Stealth) + { + if (!includeStealth && client.LoggedinUser.Stealth) + continue; + if (!client.LoggedinUser.Stealth) count++; - } + } + return count; } diff --git a/Horse Isle Server/Horse Isle Server/User.cs b/Horse Isle Server/Horse Isle Server/User.cs index d588d3d..1cf28cc 100644 --- a/Horse Isle Server/Horse Isle Server/User.cs +++ b/Horse Isle Server/Horse Isle Server/User.cs @@ -10,6 +10,7 @@ namespace Horse_Isle_Server public bool Administrator; public bool Moderator; public bool NewPlayer = false; + public Client LoggedinClient; public bool MuteAds = false; public bool MuteGlobal = false; @@ -148,7 +149,7 @@ namespace Horse_Isle_Server return SecCode; } - public User(int UserId) + public User(Client baseClient, int UserId) { if (!Database.CheckUserExist(UserId)) throw new KeyNotFoundException("User " + UserId + " not found in database!"); @@ -187,10 +188,11 @@ namespace Horse_Isle_Server SecCodeSeeds[2] = (byte)rng.Next(40, 140); SecCodeInc = (byte)rng.Next(0, 99); - // Make some friends! (Get a life!) Friends = new Friends(this); + + LoggedinClient = baseClient; } } }