| | 1 | | using System; |
| | 2 | | using System.Collections.Generic; |
| | 3 | | using System.Threading; |
| | 4 | | using Cysharp.Threading.Tasks; |
| | 5 | | using DCL.Interface; |
| | 6 | | using SocialFeaturesAnalytics; |
| | 7 | | using UnityEngine; |
| | 8 | | using Channel = DCL.Chat.Channels.Channel; |
| | 9 | |
|
| | 10 | | namespace DCL.Chat.HUD |
| | 11 | | { |
| | 12 | | public class ChatChannelHUDController : IHUD |
| | 13 | | { |
| | 14 | | private const int INITIAL_PAGE_SIZE = 30; |
| | 15 | | private const int SHOW_MORE_PAGE_SIZE = 10; |
| | 16 | | private const float REQUEST_MESSAGES_TIME_OUT = 2; |
| | 17 | |
|
| 0 | 18 | | public IChatChannelWindowView View { get; private set; } |
| | 19 | |
|
| | 20 | | private readonly DataStore dataStore; |
| 6 | 21 | | private BaseVariable<HashSet<string>> visibleTaskbarPanels => dataStore.HUDs.visibleTaskbarPanels; |
| | 22 | | private readonly IUserProfileBridge userProfileBridge; |
| | 23 | | private readonly IChatController chatController; |
| | 24 | | private readonly IMouseCatcher mouseCatcher; |
| | 25 | | private readonly InputAction_Trigger toggleChatTrigger; |
| | 26 | | private readonly ISocialAnalytics socialAnalytics; |
| | 27 | | private readonly IProfanityFilter profanityFilter; |
| | 28 | | private ChatHUDController chatHudController; |
| | 29 | | private ChannelMembersHUDController channelMembersHUDController; |
| 7 | 30 | | private CancellationTokenSource hideLoadingCancellationToken = new CancellationTokenSource(); |
| | 31 | | private bool skipChatInputTrigger; |
| | 32 | | private float lastRequestTime; |
| | 33 | | private string channelId; |
| | 34 | | private Channel channel; |
| | 35 | | private ChatMessage oldestMessage; |
| | 36 | |
|
| | 37 | | public event Action OnPressBack; |
| | 38 | | public event Action OnClosed; |
| | 39 | | public event Action<string> OnOpenChannelLeave; |
| | 40 | |
|
| 7 | 41 | | public ChatChannelHUDController(DataStore dataStore, |
| | 42 | | IUserProfileBridge userProfileBridge, |
| | 43 | | IChatController chatController, |
| | 44 | | IMouseCatcher mouseCatcher, |
| | 45 | | InputAction_Trigger toggleChatTrigger, |
| | 46 | | ISocialAnalytics socialAnalytics, |
| | 47 | | IProfanityFilter profanityFilter) |
| | 48 | | { |
| 7 | 49 | | this.dataStore = dataStore; |
| 7 | 50 | | this.userProfileBridge = userProfileBridge; |
| 7 | 51 | | this.chatController = chatController; |
| 7 | 52 | | this.mouseCatcher = mouseCatcher; |
| 7 | 53 | | this.toggleChatTrigger = toggleChatTrigger; |
| 7 | 54 | | this.socialAnalytics = socialAnalytics; |
| 7 | 55 | | this.profanityFilter = profanityFilter; |
| 7 | 56 | | } |
| | 57 | |
|
| | 58 | | public void Initialize(IChatChannelWindowView view = null) |
| | 59 | | { |
| 7 | 60 | | view ??= ChatChannelComponentView.Create(); |
| 7 | 61 | | View = view; |
| 7 | 62 | | view.OnBack -= HandlePressBack; |
| 7 | 63 | | view.OnBack += HandlePressBack; |
| 7 | 64 | | view.OnClose -= Hide; |
| 7 | 65 | | view.OnClose += Hide; |
| 7 | 66 | | view.OnRequireMoreMessages += RequestOldConversations; |
| 7 | 67 | | view.OnLeaveChannel += LeaveChannel; |
| 7 | 68 | | view.OnShowMembersList += ShowMembersList; |
| 7 | 69 | | view.OnHideMembersList += HideMembersList; |
| 7 | 70 | | view.OnMuteChanged += MuteChannel; |
| | 71 | |
|
| 7 | 72 | | chatHudController = new ChatHUDController(dataStore, userProfileBridge, false, profanityFilter); |
| 7 | 73 | | chatHudController.Initialize(view.ChatHUD); |
| 7 | 74 | | chatHudController.OnSendMessage += HandleSendChatMessage; |
| 7 | 75 | | chatHudController.OnMessageSentBlockedBySpam += HandleMessageBlockedBySpam; |
| | 76 | |
|
| 7 | 77 | | if (mouseCatcher != null) |
| 7 | 78 | | mouseCatcher.OnMouseLock += Hide; |
| | 79 | |
|
| 7 | 80 | | toggleChatTrigger.OnTriggered += HandleChatInputTriggered; |
| | 81 | |
|
| 7 | 82 | | channelMembersHUDController = new ChannelMembersHUDController(view.ChannelMembersHUD, chatController, userPr |
| 7 | 83 | | } |
| | 84 | |
|
| | 85 | | public void Setup(string channelId) |
| | 86 | | { |
| 6 | 87 | | channelMembersHUDController.SetChannelId(channelId); |
| 6 | 88 | | this.channelId = channelId; |
| 6 | 89 | | lastRequestTime = 0; |
| | 90 | |
|
| 6 | 91 | | channel = chatController.GetAllocatedChannel(channelId); |
| 6 | 92 | | View.Setup(ToPublicChatModel(channel)); |
| | 93 | |
|
| 6 | 94 | | chatHudController.ClearAllEntries(); |
| 6 | 95 | | oldestMessage = null; |
| 6 | 96 | | } |
| | 97 | |
|
| | 98 | | public void SetVisibility(bool visible) |
| | 99 | | { |
| 3 | 100 | | SetVisiblePanelList(visible); |
| | 101 | |
|
| 3 | 102 | | if (visible) |
| | 103 | | { |
| 2 | 104 | | ClearChatControllerListeners(); |
| | 105 | |
|
| 2 | 106 | | chatController.OnAddMessage += HandleMessageReceived; |
| 2 | 107 | | chatController.OnChannelLeft += HandleChannelLeft; |
| 2 | 108 | | chatController.OnChannelUpdated += HandleChannelUpdated; |
| | 109 | |
|
| 2 | 110 | | if (channelMembersHUDController.IsVisible) |
| 0 | 111 | | channelMembersHUDController.SetAutomaticReloadingActive(true); |
| | 112 | |
|
| 2 | 113 | | View?.SetLoadingMessagesActive(false); |
| 2 | 114 | | View?.SetOldMessagesLoadingActive(false); |
| | 115 | |
|
| 2 | 116 | | if (!string.IsNullOrEmpty(channelId)) |
| | 117 | | { |
| 1 | 118 | | var channel = chatController.GetAllocatedChannel(channelId); |
| 1 | 119 | | View.Setup(ToPublicChatModel(channel)); |
| | 120 | |
|
| 1 | 121 | | RequestMessages( |
| | 122 | | channelId, |
| | 123 | | INITIAL_PAGE_SIZE); |
| | 124 | | } |
| | 125 | |
|
| 2 | 126 | | View.Show(); |
| 2 | 127 | | Focus(); |
| 2 | 128 | | } |
| | 129 | | else |
| | 130 | | { |
| 1 | 131 | | ClearChatControllerListeners(); |
| | 132 | |
|
| 1 | 133 | | channelMembersHUDController.SetAutomaticReloadingActive(false); |
| 1 | 134 | | chatHudController.UnfocusInputField(); |
| 1 | 135 | | OnClosed?.Invoke(); |
| 1 | 136 | | View.Hide(); |
| | 137 | | } |
| | 138 | |
|
| 3 | 139 | | dataStore.channels.channelToBeOpenedFromLink.Set(null, notifyEvent: false); |
| 3 | 140 | | } |
| | 141 | |
|
| | 142 | | public void Dispose() |
| | 143 | | { |
| 7 | 144 | | ClearChatControllerListeners(); |
| | 145 | |
|
| 7 | 146 | | if (mouseCatcher != null) |
| 7 | 147 | | mouseCatcher.OnMouseLock -= Hide; |
| | 148 | |
|
| 7 | 149 | | toggleChatTrigger.OnTriggered -= HandleChatInputTriggered; |
| | 150 | |
|
| 7 | 151 | | chatHudController.OnSendMessage -= HandleSendChatMessage; |
| 7 | 152 | | chatHudController.OnMessageSentBlockedBySpam -= HandleMessageBlockedBySpam; |
| | 153 | |
|
| 7 | 154 | | if (View != null) |
| | 155 | | { |
| 7 | 156 | | View.OnBack -= HandlePressBack; |
| 7 | 157 | | View.OnClose -= Hide; |
| 7 | 158 | | View.OnRequireMoreMessages -= RequestOldConversations; |
| 7 | 159 | | View.OnLeaveChannel -= LeaveChannel; |
| 7 | 160 | | View.OnMuteChanged -= MuteChannel; |
| 7 | 161 | | View.Dispose(); |
| | 162 | | } |
| | 163 | |
|
| 7 | 164 | | hideLoadingCancellationToken.Dispose(); |
| 7 | 165 | | channelMembersHUDController.Dispose(); |
| 7 | 166 | | } |
| | 167 | |
|
| | 168 | | private void HandleSendChatMessage(ChatMessage message) |
| | 169 | | { |
| 2 | 170 | | message.messageType = ChatMessage.Type.PUBLIC; |
| 2 | 171 | | message.recipient = channelId; |
| | 172 | |
|
| 2 | 173 | | var isValidMessage = !string.IsNullOrEmpty(message.body) |
| | 174 | | && !string.IsNullOrWhiteSpace(message.body) |
| | 175 | | && !string.IsNullOrEmpty(message.recipient); |
| | 176 | |
|
| 2 | 177 | | if (isValidMessage) |
| | 178 | | { |
| 2 | 179 | | chatHudController.ResetInputField(); |
| 2 | 180 | | chatHudController.FocusInputField(); |
| 2 | 181 | | } |
| | 182 | | else |
| | 183 | | { |
| 0 | 184 | | SetVisibility(false); |
| 0 | 185 | | return; |
| | 186 | | } |
| | 187 | |
|
| 2 | 188 | | if (message.body.ToLower().Equals("/leave")) |
| | 189 | | { |
| 1 | 190 | | LeaveChannelFromCommand(); |
| 1 | 191 | | return; |
| | 192 | | } |
| | 193 | |
|
| 1 | 194 | | chatController.Send(message); |
| 1 | 195 | | socialAnalytics.SendMessageSentToChannel(channel.Name, message.body.Length, "channel"); |
| 1 | 196 | | } |
| | 197 | |
|
| | 198 | | private void HandleMessageReceived(ChatMessage message) |
| | 199 | | { |
| 0 | 200 | | if (!IsMessageFomCurrentChannel(message)) return; |
| | 201 | |
|
| 0 | 202 | | UpdateOldestMessage(message); |
| | 203 | |
|
| 0 | 204 | | message.isChannelMessage = true; |
| | 205 | | // TODO: right now the channel history is disabled, but we must find a workaround to support history + max m |
| | 206 | | // one approach could be to increment the max amount of messages depending on how many pages you loaded from |
| | 207 | | // for example: 1 page = 30 messages, 2 pages = 60 messages, and so on.. |
| 0 | 208 | | chatHudController.AddChatMessage(message, limitMaxEntries: true); |
| | 209 | |
|
| 0 | 210 | | if (View.IsActive) |
| | 211 | | { |
| | 212 | | // The messages from 'channelId' are marked as read if the channel window is currently open |
| 0 | 213 | | MarkChannelMessagesAsRead(); |
| | 214 | | } |
| | 215 | |
|
| 0 | 216 | | View?.SetLoadingMessagesActive(false); |
| 0 | 217 | | View?.SetOldMessagesLoadingActive(false); |
| 0 | 218 | | } |
| | 219 | |
|
| | 220 | | private void UpdateOldestMessage(ChatMessage message) |
| | 221 | | { |
| 0 | 222 | | if (oldestMessage == null) |
| 0 | 223 | | oldestMessage = message; |
| 0 | 224 | | else if (message.timestamp < oldestMessage.timestamp) |
| 0 | 225 | | oldestMessage = message; |
| 0 | 226 | | } |
| | 227 | |
|
| | 228 | | private void Hide() |
| | 229 | | { |
| 0 | 230 | | SetVisibility(false); |
| 0 | 231 | | OnClosed?.Invoke(); |
| 0 | 232 | | } |
| | 233 | |
|
| 0 | 234 | | private void HandlePressBack() => OnPressBack?.Invoke(); |
| | 235 | |
|
| | 236 | | private bool IsMessageFomCurrentChannel(ChatMessage message) => |
| 0 | 237 | | message.sender == channelId || message.recipient == channelId || (View.IsActive && message.messageType == Ch |
| | 238 | |
|
| 2 | 239 | | private void MarkChannelMessagesAsRead() => chatController.MarkChannelMessagesAsSeen(channelId); |
| | 240 | |
|
| | 241 | | private void HandleChatInputTriggered(DCLAction_Trigger action) |
| | 242 | | { |
| | 243 | | // race condition patch caused by unfocusing input field from invalid message on SendChatMessage |
| | 244 | | // chat input trigger is the same key as sending the chat message from the input field |
| 0 | 245 | | if (skipChatInputTrigger) |
| | 246 | | { |
| 0 | 247 | | skipChatInputTrigger = false; |
| 0 | 248 | | return; |
| | 249 | | } |
| | 250 | |
|
| 0 | 251 | | if (!View.IsActive) return; |
| 0 | 252 | | chatHudController.FocusInputField(); |
| 0 | 253 | | } |
| | 254 | |
|
| | 255 | | private void RequestMessages(string channelId, int limit, string fromMessageId = null) |
| | 256 | | { |
| 1 | 257 | | View?.SetLoadingMessagesActive(true); |
| 1 | 258 | | chatController.GetChannelMessages(channelId, limit, fromMessageId); |
| 1 | 259 | | hideLoadingCancellationToken.Cancel(); |
| 1 | 260 | | hideLoadingCancellationToken = new CancellationTokenSource(); |
| 1 | 261 | | WaitForRequestTimeOutThenHideLoadingFeedback(hideLoadingCancellationToken.Token).Forget(); |
| 1 | 262 | | } |
| | 263 | |
|
| | 264 | | private void RequestOldConversations() |
| | 265 | | { |
| 0 | 266 | | if (IsLoadingMessages()) return; |
| | 267 | |
|
| 0 | 268 | | View?.SetOldMessagesLoadingActive(true); |
| 0 | 269 | | lastRequestTime = Time.realtimeSinceStartup; |
| | 270 | |
|
| 0 | 271 | | chatController.GetChannelMessages( |
| | 272 | | channelId, |
| | 273 | | SHOW_MORE_PAGE_SIZE, |
| | 274 | | oldestMessage?.messageId); |
| | 275 | |
|
| 0 | 276 | | hideLoadingCancellationToken.Cancel(); |
| 0 | 277 | | hideLoadingCancellationToken = new CancellationTokenSource(); |
| 0 | 278 | | WaitForRequestTimeOutThenHideLoadingFeedback(hideLoadingCancellationToken.Token).Forget(); |
| 0 | 279 | | } |
| | 280 | |
|
| | 281 | | private bool IsLoadingMessages() => |
| 0 | 282 | | Time.realtimeSinceStartup - lastRequestTime < REQUEST_MESSAGES_TIME_OUT; |
| | 283 | |
|
| | 284 | | private async UniTaskVoid WaitForRequestTimeOutThenHideLoadingFeedback(CancellationToken cancellationToken) |
| | 285 | | { |
| 1 | 286 | | lastRequestTime = Time.realtimeSinceStartup; |
| | 287 | |
|
| 3 | 288 | | await UniTask.WaitUntil(() => |
| 284 | 289 | | Time.realtimeSinceStartup - lastRequestTime > REQUEST_MESSAGES_TIME_OUT, |
| | 290 | | cancellationToken: cancellationToken); |
| 1 | 291 | | if (cancellationToken.IsCancellationRequested) return; |
| | 292 | |
|
| 1 | 293 | | View?.SetLoadingMessagesActive(false); |
| 1 | 294 | | View?.SetOldMessagesLoadingActive(false); |
| 1 | 295 | | } |
| | 296 | |
|
| | 297 | | private void LeaveChannel() |
| | 298 | | { |
| 1 | 299 | | dataStore.channels.channelLeaveSource.Set(ChannelLeaveSource.Chat); |
| 1 | 300 | | OnOpenChannelLeave?.Invoke(channelId); |
| 1 | 301 | | } |
| | 302 | |
|
| | 303 | | private void LeaveChannelFromCommand() |
| | 304 | | { |
| 1 | 305 | | dataStore.channels.channelLeaveSource.Set(ChannelLeaveSource.Command); |
| 1 | 306 | | chatController.LeaveChannel(channelId); |
| 1 | 307 | | } |
| | 308 | |
|
| | 309 | | private void HandleChannelLeft(string channelId) |
| | 310 | | { |
| 1 | 311 | | if (channelId != this.channelId) return; |
| 1 | 312 | | OnPressBack?.Invoke(); |
| 1 | 313 | | } |
| | 314 | |
|
| | 315 | | private void HandleChannelUpdated(Channel updatedChannel) |
| | 316 | | { |
| 0 | 317 | | if (updatedChannel.ChannelId != channelId) |
| 0 | 318 | | return; |
| | 319 | |
|
| 0 | 320 | | View.Setup(ToPublicChatModel(updatedChannel)); |
| 0 | 321 | | channelMembersHUDController.SetMembersCount(updatedChannel.MemberCount); |
| 0 | 322 | | } |
| | 323 | |
|
| 0 | 324 | | private void ShowMembersList() => channelMembersHUDController.SetVisibility(true); |
| | 325 | |
|
| 0 | 326 | | private void HideMembersList() => channelMembersHUDController.SetVisibility(false); |
| | 327 | |
|
| | 328 | | private void MuteChannel(bool muted) |
| | 329 | | { |
| 2 | 330 | | if (muted) |
| 1 | 331 | | chatController.MuteChannel(channelId); |
| | 332 | | else |
| 1 | 333 | | chatController.UnmuteChannel(channelId); |
| 1 | 334 | | } |
| | 335 | |
|
| | 336 | | private void SetVisiblePanelList(bool visible) |
| | 337 | | { |
| 3 | 338 | | var newSet = visibleTaskbarPanels.Get(); |
| | 339 | |
|
| 3 | 340 | | if (visible) |
| 2 | 341 | | newSet.Add("ChatChannel"); |
| | 342 | | else |
| 1 | 343 | | newSet.Remove("ChatChannel"); |
| | 344 | |
|
| 3 | 345 | | visibleTaskbarPanels.Set(newSet, true); |
| 3 | 346 | | } |
| | 347 | |
|
| | 348 | | private PublicChatModel ToPublicChatModel(Channel channel) |
| | 349 | | { |
| 7 | 350 | | return new PublicChatModel(channelId, channel.Name, channel.Description, |
| | 351 | | channel.Joined, channel.MemberCount, channel.Muted); |
| | 352 | | } |
| | 353 | |
|
| | 354 | | private void ClearChatControllerListeners() |
| | 355 | | { |
| 10 | 356 | | if (chatController == null) return; |
| 10 | 357 | | chatController.OnAddMessage -= HandleMessageReceived; |
| 10 | 358 | | chatController.OnChannelLeft -= HandleChannelLeft; |
| 10 | 359 | | chatController.OnChannelUpdated -= HandleChannelUpdated; |
| 10 | 360 | | } |
| | 361 | |
|
| | 362 | | private void Focus() |
| | 363 | | { |
| 2 | 364 | | chatHudController.FocusInputField(); |
| 2 | 365 | | MarkChannelMessagesAsRead(); |
| 2 | 366 | | } |
| | 367 | |
|
| | 368 | | private void HandleMessageBlockedBySpam(ChatMessage message) |
| | 369 | | { |
| 0 | 370 | | chatHudController.AddChatMessage(new ChatEntryModel |
| | 371 | | { |
| | 372 | | timestamp = (ulong) DateTimeOffset.UtcNow.ToUnixTimeMilliseconds(), |
| | 373 | | bodyText = "You sent too many messages in a short period of time. Please wait and try again later.", |
| | 374 | | messageId = Guid.NewGuid().ToString(), |
| | 375 | | messageType = ChatMessage.Type.SYSTEM, |
| | 376 | | subType = ChatEntryModel.SubType.RECEIVED |
| | 377 | | }); |
| 0 | 378 | | } |
| | 379 | | } |
| | 380 | | } |