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