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