Messenger Tüm Mesajlar Logluyor

This commit is contained in:
Sedat Öztürk 2026-06-12 22:35:48 +03:00
parent 7632c9e8a0
commit af726c530a
17 changed files with 1196 additions and 92 deletions

View file

@ -3,6 +3,7 @@
using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;
using Volo.Abp.Application.Dtos;
using Volo.Abp.Content;
namespace Sozsoft.Platform.Messenger;
@ -36,6 +37,8 @@ public class MessengerUploadAttachmentInput
public class MessengerSendMessageDto
{
public Guid? ConversationId { get; set; }
[Required]
public List<Guid> RecipientIds { get; set; } = new();
@ -49,6 +52,7 @@ public class MessengerMessageDto
{
public Guid Id { get; set; }
public Guid? TenantId { get; set; }
public Guid ConversationId { get; set; }
public Guid SenderId { get; set; }
public string SenderUserName { get; set; } = string.Empty;
public string SenderName { get; set; } = string.Empty;
@ -57,3 +61,43 @@ public class MessengerMessageDto
public List<MessengerAttachmentDto> Attachments { get; set; } = new();
public DateTime SentAt { get; set; }
}
public class MessengerMessageDeletedDto
{
public Guid MessageId { get; set; }
public Guid ConversationId { get; set; }
public Guid SenderId { get; set; }
public List<Guid> RecipientIds { get; set; } = new();
}
public class MessengerConversationDto : FullAuditedEntityDto<Guid>
{
public Guid? TenantId { get; set; }
public string? Title { get; set; }
public List<Guid> ParticipantIds { get; set; } = new();
public bool IsGroup { get; set; }
public Guid? LastSenderId { get; set; }
public string? LastMessagePreview { get; set; }
public DateTime? LastMessageTime { get; set; }
public int MessageCount { get; set; }
}
public class MessengerConversationCreateUpdateDto
{
[StringLength(256)]
public string? Title { get; set; }
[Required]
public List<Guid> ParticipantIds { get; set; } = new();
}
public class MessengerConversationListRequestDto : PagedAndSortedResultRequestDto
{
public string? Filter { get; set; }
}
public class MessengerGetMessagesInput : PagedAndSortedResultRequestDto
{
public Guid? ConversationId { get; set; }
public List<Guid> ParticipantIds { get; set; } = new();
}

View file

@ -4,12 +4,16 @@ using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Linq.Dynamic.Core;
using System.Text.Json;
using System.Threading.Tasks;
using Sozsoft.Platform.BlobStoring;
using Sozsoft.Platform.Entities;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Mvc;
using Microsoft.Extensions.Configuration;
using Volo.Abp;
using Volo.Abp.Application.Dtos;
using Volo.Abp.Application.Services;
using Volo.Abp.Domain.Repositories;
using Volo.Abp.Identity;
@ -20,15 +24,21 @@ namespace Sozsoft.Platform.Messenger;
public class MessengerAppService : ApplicationService
{
private readonly IRepository<IdentityUser, Guid> _userRepository;
private readonly IRepository<MessengerConversation, Guid> _conversationRepository;
private readonly IRepository<MessengerConversationMessage, Guid> _messageRepository;
private readonly BlobManager _blobManager;
private readonly IConfiguration _configuration;
public MessengerAppService(
IRepository<IdentityUser, Guid> userRepository,
IRepository<MessengerConversation, Guid> conversationRepository,
IRepository<MessengerConversationMessage, Guid> messageRepository,
BlobManager blobManager,
IConfiguration configuration)
{
_userRepository = userRepository;
_conversationRepository = conversationRepository;
_messageRepository = messageRepository;
_blobManager = blobManager;
_configuration = configuration;
}
@ -78,6 +88,250 @@ public class MessengerAppService : ApplicationService
}).ToList();
}
public async Task<PagedResultDto<MessengerConversationDto>> GetConversationsAsync(MessengerConversationListRequestDto input)
{
var query = await _conversationRepository.GetQueryableAsync();
var currentUserId = GetCurrentUserId();
var currentUserKey = currentUserId.ToString("N");
query = query.Where(conversation => conversation.ParticipantKey.Contains(currentUserKey));
if (!input.Filter.IsNullOrWhiteSpace())
{
var filter = input.Filter!.Trim().ToLower();
query = query.Where(conversation =>
(conversation.Title != null && conversation.Title.ToLower().Contains(filter)) ||
(conversation.LastMessagePreview != null && conversation.LastMessagePreview.ToLower().Contains(filter)));
}
var totalCount = await AsyncExecuter.CountAsync(query);
var sorting = input.Sorting.IsNullOrWhiteSpace()
? $"{nameof(MessengerConversation.LastMessageTime)} desc"
: input.Sorting;
var conversations = await AsyncExecuter.ToListAsync(
query.OrderBy(sorting).Skip(input.SkipCount).Take(input.MaxResultCount)
);
return new PagedResultDto<MessengerConversationDto>(
totalCount,
conversations.Select(MapConversation).ToList()
);
}
public async Task<MessengerConversationDto> GetConversationAsync(Guid id)
{
var conversation = await _conversationRepository.GetAsync(id);
EnsureConversationParticipant(conversation, GetCurrentUserId());
return MapConversation(conversation);
}
public async Task<MessengerConversationDto> CreateConversationAsync(MessengerConversationCreateUpdateDto input)
{
var currentUserId = GetCurrentUserId();
var participantIds = await NormalizeAndValidateParticipantIdsAsync(input.ParticipantIds, currentUserId);
var participantKey = CreateParticipantKey(participantIds);
var existing = await FindConversationByParticipantKeyAsync(participantKey);
if (existing is not null)
{
return MapConversation(existing);
}
var conversation = new MessengerConversation(GuidGenerator.Create())
{
TenantId = CurrentTenant.Id,
Title = input.Title?.Trim(),
ParticipantKey = participantKey,
ParticipantIdsJson = JsonSerializer.Serialize(participantIds),
IsGroup = participantIds.Count > 2
};
await _conversationRepository.InsertAsync(conversation, autoSave: true);
return MapConversation(conversation);
}
public async Task<MessengerConversationDto> UpdateConversationAsync(Guid id, MessengerConversationCreateUpdateDto input)
{
var conversation = await _conversationRepository.GetAsync(id);
var currentUserId = GetCurrentUserId();
EnsureConversationParticipant(conversation, currentUserId);
var participantIds = await NormalizeAndValidateParticipantIdsAsync(input.ParticipantIds, currentUserId);
var participantKey = CreateParticipantKey(participantIds);
conversation.Title = input.Title?.Trim();
conversation.ParticipantKey = participantKey;
conversation.ParticipantIdsJson = JsonSerializer.Serialize(participantIds);
conversation.IsGroup = participantIds.Count > 2;
await _conversationRepository.UpdateAsync(conversation, autoSave: true);
return MapConversation(conversation);
}
public async Task DeleteConversationAsync(Guid id)
{
var conversation = await _conversationRepository.GetAsync(id);
EnsureConversationParticipant(conversation, GetCurrentUserId());
await _conversationRepository.DeleteAsync(conversation, autoSave: true);
}
[HttpPost("api/app/messenger/messages")]
public async Task<List<MessengerMessageDto>> GetMessagesAsync(MessengerGetMessagesInput input)
{
var currentUserId = GetCurrentUserId();
var conversation = input.ConversationId.HasValue
? await _conversationRepository.GetAsync(input.ConversationId.Value)
: await FindConversationByParticipantKeyAsync(
CreateParticipantKey(await NormalizeAndValidateParticipantIdsAsync(input.ParticipantIds, currentUserId))
);
if (conversation is null)
{
return new List<MessengerMessageDto>();
}
EnsureConversationParticipant(conversation, currentUserId);
var query = await _messageRepository.GetQueryableAsync();
query = query.Where(message => message.ConversationId == conversation.Id);
var sorting = input.Sorting.IsNullOrWhiteSpace()
? $"{nameof(MessengerConversationMessage.SentAt)} desc"
: input.Sorting;
var messages = await AsyncExecuter.ToListAsync(
query.OrderBy(sorting).Skip(input.SkipCount).Take(input.MaxResultCount)
);
return messages
.OrderBy(message => message.SentAt)
.Select(MapMessage)
.ToList();
}
public async Task<MessengerMessageDto> SendMessageAsync(MessengerSendMessageDto input)
{
var senderId = GetCurrentUserId();
var recipientIds = input.RecipientIds
.Where(id => id != Guid.Empty && id != senderId)
.Distinct()
.ToList();
if (recipientIds.Count == 0)
{
throw new UserFriendlyException("En az bir alici secilmelidir.");
}
if (input.Text.IsNullOrWhiteSpace() && input.Attachments.Count == 0)
{
throw new UserFriendlyException("Mesaj veya dosya gonderilmelidir.");
}
var validRecipients = await GetValidUsersAsync(recipientIds);
if (validRecipients.Count != recipientIds.Count)
{
throw new UserFriendlyException("Bu tenant icinde gecersiz alici var.");
}
MessengerConversation? conversation = null;
if (input.ConversationId.HasValue)
{
conversation = await _conversationRepository.GetAsync(input.ConversationId.Value);
EnsureConversationParticipant(conversation, senderId);
}
var participantIds = recipientIds.Append(senderId).Distinct().ToList();
var participantKey = CreateParticipantKey(participantIds);
conversation ??= await FindConversationByParticipantKeyAsync(participantKey);
if (conversation is null)
{
conversation = new MessengerConversation(GuidGenerator.Create())
{
TenantId = CurrentTenant.Id,
ParticipantKey = participantKey,
ParticipantIdsJson = JsonSerializer.Serialize(participantIds),
IsGroup = participantIds.Count > 2
};
await _conversationRepository.InsertAsync(conversation, autoSave: true);
}
var senderName = $"{CurrentUser.Name} {CurrentUser.SurName}".Trim() ?? CurrentUser.UserName ?? "Kullanici";
var trimmedText = input.Text?.Trim();
var message = new MessengerConversationMessage(GuidGenerator.Create())
{
TenantId = CurrentTenant.Id,
ConversationId = conversation.Id,
SenderId = senderId,
SenderUserName = CurrentUser.UserName ?? string.Empty,
SenderName = senderName,
RecipientIdsJson = JsonSerializer.Serialize(recipientIds),
Text = trimmedText,
AttachmentsJson = JsonSerializer.Serialize(input.Attachments),
SentAt = Clock.Now
};
await _messageRepository.InsertAsync(message, autoSave: true);
conversation.LastSenderId = senderId;
conversation.LastMessagePreview = GetMessagePreview(trimmedText, input.Attachments.Count);
conversation.LastMessageTime = message.SentAt;
conversation.MessageCount += 1;
await _conversationRepository.UpdateAsync(conversation, autoSave: true);
return MapMessage(message);
}
public async Task<MessengerMessageDeletedDto> DeleteMessageAsync(Guid id)
{
var currentUserId = GetCurrentUserId();
var message = await _messageRepository.GetAsync(id);
if (message.SenderId != currentUserId)
{
throw new UserFriendlyException("Sadece kendi mesajlarinizi silebilirsiniz.");
}
if (Clock.Now - message.SentAt > TimeSpan.FromMinutes(10))
{
throw new UserFriendlyException("Mesajlar ilk 10 dakika icinde silinebilir.");
}
var recipientIds = DeserializeGuidList(message.RecipientIdsJson);
var conversation = await _conversationRepository.GetAsync(message.ConversationId);
await _messageRepository.DeleteAsync(message, autoSave: true);
var query = await _messageRepository.GetQueryableAsync();
var lastMessage = await AsyncExecuter.FirstOrDefaultAsync(
query
.Where(item => item.ConversationId == conversation.Id)
.OrderByDescending(item => item.SentAt)
);
conversation.MessageCount = Math.Max(0, conversation.MessageCount - 1);
conversation.LastSenderId = lastMessage?.SenderId;
conversation.LastMessagePreview = lastMessage is null
? null
: GetMessagePreview(lastMessage.Text, DeserializeAttachments(lastMessage.AttachmentsJson).Count);
conversation.LastMessageTime = lastMessage?.SentAt;
await _conversationRepository.UpdateAsync(conversation, autoSave: true);
return new MessengerMessageDeletedDto
{
MessageId = message.Id,
ConversationId = message.ConversationId,
SenderId = message.SenderId,
RecipientIds = recipientIds
};
}
[HttpPost("api/app/messenger/upload-attachment")]
public async Task<MessengerAttachmentDto> UploadAttachmentAsync([FromForm] MessengerUploadAttachmentInput input)
{
@ -111,4 +365,131 @@ public class MessengerAppService : ApplicationService
Url = $"{baseUrl}/{tenantPart}/{BlobContainerNames.Messenger}/{savedFileName}"
};
}
private Guid GetCurrentUserId()
{
if (!CurrentUser.Id.HasValue)
{
throw new UserFriendlyException("Kullanici oturumu bulunamadi.");
}
return CurrentUser.Id.Value;
}
private async Task<List<Guid>> NormalizeAndValidateParticipantIdsAsync(IEnumerable<Guid> participantIds, Guid currentUserId)
{
var normalizedIds = participantIds
.Where(id => id != Guid.Empty)
.Append(currentUserId)
.Distinct()
.ToList();
if (normalizedIds.Count < 2)
{
throw new UserFriendlyException("En az iki katilimci secilmelidir.");
}
var users = await GetValidUsersAsync(normalizedIds);
if (users.Count != normalizedIds.Count)
{
throw new UserFriendlyException("Bu tenant icinde gecersiz katilimci var.");
}
return normalizedIds;
}
private async Task<List<IdentityUser>> GetValidUsersAsync(List<Guid> userIds)
{
return await _userRepository.GetListAsync(user =>
userIds.Contains(user.Id) &&
user.TenantId == CurrentTenant.Id &&
user.IsActive);
}
private async Task<MessengerConversation?> FindConversationByParticipantKeyAsync(string participantKey)
{
var query = await _conversationRepository.GetQueryableAsync();
return await AsyncExecuter.FirstOrDefaultAsync(query.Where(conversation => conversation.ParticipantKey == participantKey));
}
private static string CreateParticipantKey(IEnumerable<Guid> participantIds)
{
return string.Join(",", participantIds.Select(id => id.ToString("N")).OrderBy(id => id));
}
private static void EnsureConversationParticipant(MessengerConversation conversation, Guid currentUserId)
{
var participantIds = DeserializeGuidList(conversation.ParticipantIdsJson);
if (!participantIds.Contains(currentUserId))
{
throw new UserFriendlyException("Bu gorusmeye erisim yetkiniz yok.");
}
}
private static MessengerConversationDto MapConversation(MessengerConversation conversation)
{
return new MessengerConversationDto
{
Id = conversation.Id,
TenantId = conversation.TenantId,
Title = conversation.Title,
ParticipantIds = DeserializeGuidList(conversation.ParticipantIdsJson),
IsGroup = conversation.IsGroup,
LastSenderId = conversation.LastSenderId,
LastMessagePreview = conversation.LastMessagePreview,
LastMessageTime = conversation.LastMessageTime,
MessageCount = conversation.MessageCount,
CreationTime = conversation.CreationTime,
CreatorId = conversation.CreatorId,
LastModificationTime = conversation.LastModificationTime,
LastModifierId = conversation.LastModifierId
};
}
private static MessengerMessageDto MapMessage(MessengerConversationMessage message)
{
return new MessengerMessageDto
{
Id = message.Id,
TenantId = message.TenantId,
ConversationId = message.ConversationId,
SenderId = message.SenderId,
SenderUserName = message.SenderUserName,
SenderName = message.SenderName,
RecipientIds = DeserializeGuidList(message.RecipientIdsJson),
Text = message.Text,
Attachments = DeserializeAttachments(message.AttachmentsJson),
SentAt = message.SentAt
};
}
private static List<Guid> DeserializeGuidList(string? json)
{
if (json.IsNullOrWhiteSpace())
{
return new List<Guid>();
}
return JsonSerializer.Deserialize<List<Guid>>(json!) ?? new List<Guid>();
}
private static List<MessengerAttachmentDto> DeserializeAttachments(string? json)
{
if (json.IsNullOrWhiteSpace())
{
return new List<MessengerAttachmentDto>();
}
return JsonSerializer.Deserialize<List<MessengerAttachmentDto>>(json!) ?? new List<MessengerAttachmentDto>();
}
private static string GetMessagePreview(string? text, int attachmentCount)
{
if (!text.IsNullOrWhiteSpace())
{
return text!.Length > 512 ? text[..512] : text;
}
return attachmentCount > 0 ? $"{attachmentCount} dosya" : string.Empty;
}
}

View file

@ -5,9 +5,6 @@ using System.Linq;
using System.Threading.Tasks;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.SignalR;
using Volo.Abp.Domain.Repositories;
using Volo.Abp.Guids;
using Volo.Abp.Identity;
using Volo.Abp.MultiTenancy;
using Volo.Abp.Users;
@ -18,19 +15,16 @@ public class MessengerHub : Hub
{
private readonly ICurrentTenant _currentTenant;
private readonly ICurrentUser _currentUser;
private readonly IGuidGenerator _guidGenerator;
private readonly IRepository<IdentityUser, Guid> _userRepository;
private readonly MessengerAppService _messengerAppService;
public MessengerHub(
ICurrentTenant currentTenant,
ICurrentUser currentUser,
IGuidGenerator guidGenerator,
IRepository<IdentityUser, Guid> userRepository)
MessengerAppService messengerAppService)
{
_currentTenant = currentTenant;
_currentUser = currentUser;
_guidGenerator = guidGenerator;
_userRepository = userRepository;
_messengerAppService = messengerAppService;
}
public override async Task OnConnectedAsync()
@ -57,50 +51,7 @@ public class MessengerHub : Hub
public async Task SendMessage(MessengerSendMessageDto input)
{
if (!_currentUser.Id.HasValue)
{
throw new HubException("Kullanıcı oturumu bulunamadı.");
}
var recipientIds = input.RecipientIds
.Where(id => id != Guid.Empty)
.Distinct()
.ToList();
if (recipientIds.Count == 0)
{
throw new HubException("En az bir alıcı seçilmelidir.");
}
if (input.Text.IsNullOrWhiteSpace() && input.Attachments.Count == 0)
{
throw new HubException("Mesaj veya dosya gönderilmelidir.");
}
var tenantId = _currentTenant.Id;
var validRecipients = await _userRepository.GetListAsync(user =>
recipientIds.Contains(user.Id) &&
user.TenantId == tenantId &&
user.IsActive);
if (validRecipients.Count == 0)
{
throw new HubException("Bu tenant içinde geçerli alıcı bulunamadı.");
}
var senderName = _currentUser.Name ?? _currentUser.UserName ?? "Kullanıcı";
var message = new MessengerMessageDto
{
Id = _guidGenerator.Create(),
TenantId = tenantId,
SenderId = _currentUser.Id.Value,
SenderUserName = _currentUser.UserName ?? string.Empty,
SenderName = senderName,
RecipientIds = validRecipients.Select(user => user.Id).ToList(),
Text = input.Text?.Trim(),
Attachments = input.Attachments,
SentAt = DateTime.UtcNow
};
var message = await _messengerAppService.SendMessageAsync(input);
var targetGroups = message.RecipientIds
.Append(message.SenderId)
@ -111,6 +62,19 @@ public class MessengerHub : Hub
await Clients.Groups(targetGroups).SendAsync("MessengerMessageReceived", message);
}
public async Task DeleteMessage(Guid messageId)
{
var deletedMessage = await _messengerAppService.DeleteMessageAsync(messageId);
var targetGroups = deletedMessage.RecipientIds
.Append(deletedMessage.SenderId)
.Distinct()
.Select(UserGroupName)
.ToList();
await Clients.Groups(targetGroups).SendAsync("MessengerMessageDeleted", deletedMessage);
}
private string TenantKey => _currentTenant.Id?.ToString("N") ?? "host";
private string TenantGroupName() => $"messenger:tenant:{TenantKey}";

View file

@ -809,7 +809,7 @@ public class ListFormSeeder_Administration : IDataSeedContributor, ITransientDep
EditingOptionJson = DefaultEditingOptionJson(listFormName, 500, 710, true, true, true, true, false),
EditingFormJson = JsonSerializer.Serialize(new List<EditingFormDto>() {
new () { Order=1,ColCount=1,ColSpan=1,ItemType="group",Items=[
new EditingFormItemDto { Order=1, DataField="Avatar", ColSpan=1, EditorType2=EditorTypes.dxImageViewer, EditorOptions=EditorOptionValues.ImageUploadOptions(false) },
new EditingFormItemDto { Order=1, DataField="Avatar", ColSpan=1, EditorType2=EditorTypes.dxImageUpload, EditorOptions=EditorOptionValues.ImageUploadOptions(false) },
new EditingFormItemDto { Order=2, DataField="Email", ColSpan=1, EditorType2=EditorTypes.dxTextBox },
new EditingFormItemDto { Order=3, DataField="Name", ColSpan=1, EditorType2=EditorTypes.dxTextBox },
new EditingFormItemDto { Order=4, DataField="Surname", ColSpan=1, EditorType2=EditorTypes.dxTextBox },

View file

@ -85,6 +85,8 @@ public enum TableNameEnum
Event,
EventPhoto,
EventComment,
MessengerConversation,
MessengerConversationMessage,
Videoroom,
VideoroomParticipant,
VideoroomAttandance,

View file

@ -104,6 +104,8 @@ public static class TableNameResolver
{ nameof(TableNameEnum.Event), (TablePrefix.TenantByName, MenuPrefix.Administration) },
{ nameof(TableNameEnum.EventPhoto), (TablePrefix.TenantByName, MenuPrefix.Administration) },
{ nameof(TableNameEnum.EventComment), (TablePrefix.TenantByName, MenuPrefix.Administration) },
{ nameof(TableNameEnum.MessengerConversation), (TablePrefix.TenantByName, MenuPrefix.Administration) },
{ nameof(TableNameEnum.MessengerConversationMessage), (TablePrefix.TenantByName, MenuPrefix.Administration) },
};

View file

@ -0,0 +1,29 @@
using System;
using System.Collections.Generic;
using Volo.Abp.Domain.Entities.Auditing;
using Volo.Abp.MultiTenancy;
namespace Sozsoft.Platform.Entities;
public class MessengerConversation : FullAuditedEntity<Guid>, IMultiTenant
{
protected MessengerConversation()
{
}
public MessengerConversation(Guid id) : base(id)
{
}
public Guid? TenantId { get; set; }
public string? Title { get; set; }
public string ParticipantKey { get; set; } = string.Empty;
public string ParticipantIdsJson { get; set; } = "[]";
public bool IsGroup { get; set; }
public Guid? LastSenderId { get; set; }
public string? LastMessagePreview { get; set; }
public DateTime? LastMessageTime { get; set; }
public int MessageCount { get; set; }
public virtual ICollection<MessengerConversationMessage> Messages { get; set; } = new List<MessengerConversationMessage>();
}

View file

@ -0,0 +1,28 @@
using System;
using Volo.Abp.Domain.Entities.Auditing;
using Volo.Abp.MultiTenancy;
namespace Sozsoft.Platform.Entities;
public class MessengerConversationMessage : FullAuditedEntity<Guid>, IMultiTenant
{
protected MessengerConversationMessage()
{
}
public MessengerConversationMessage(Guid id) : base(id)
{
}
public Guid? TenantId { get; set; }
public Guid ConversationId { get; set; }
public Guid SenderId { get; set; }
public string SenderUserName { get; set; } = string.Empty;
public string SenderName { get; set; } = string.Empty;
public string RecipientIdsJson { get; set; } = "[]";
public string? Text { get; set; }
public string AttachmentsJson { get; set; } = "[]";
public DateTime SentAt { get; set; }
public virtual MessengerConversation Conversation { get; set; } = default!;
}

View file

@ -120,6 +120,9 @@ public class PlatformDbContext :
public DbSet<EventComment> EventComments { get; set; }
public DbSet<EventLike> EventLikes { get; set; }
public DbSet<MessengerConversation> MessengerConversations { get; set; }
public DbSet<MessengerConversationMessage> MessengerConversationMessages { get; set; }
public DbSet<Announcement> Announcements { get; set; }
public DbSet<Survey> Surveys { get; set; }
@ -1370,6 +1373,40 @@ public class PlatformDbContext :
.OnDelete(DeleteBehavior.Cascade);
});
builder.Entity<MessengerConversation>(b =>
{
b.ToTable(TableNameResolver.GetFullTableName(nameof(TableNameEnum.MessengerConversation)), Prefix.DbSchema);
b.ConfigureByConvention();
b.Property(x => x.Title).HasMaxLength(256);
b.Property(x => x.ParticipantKey).IsRequired().HasMaxLength(512);
b.Property(x => x.ParticipantIdsJson).IsRequired().HasColumnType("text");
b.Property(x => x.LastMessagePreview).HasMaxLength(512);
b.Property(x => x.MessageCount).HasDefaultValue(0);
b.HasIndex(x => new { x.TenantId, x.ParticipantKey }).IsUnique().HasFilter("[IsDeleted] = 0");
b.HasMany(x => x.Messages)
.WithOne(x => x.Conversation)
.HasForeignKey(x => x.ConversationId)
.OnDelete(DeleteBehavior.Cascade);
});
builder.Entity<MessengerConversationMessage>(b =>
{
b.ToTable(TableNameResolver.GetFullTableName(nameof(TableNameEnum.MessengerConversationMessage)), Prefix.DbSchema);
b.ConfigureByConvention();
b.Property(x => x.SenderUserName).IsRequired().HasMaxLength(256);
b.Property(x => x.SenderName).IsRequired().HasMaxLength(256);
b.Property(x => x.RecipientIdsJson).IsRequired().HasColumnType("text");
b.Property(x => x.Text).HasMaxLength(4096);
b.Property(x => x.AttachmentsJson).IsRequired().HasColumnType("text");
b.Property(x => x.SentAt).IsRequired();
b.HasIndex(x => new { x.TenantId, x.ConversationId, x.SentAt });
});
//Videoroom
builder.Entity<Videoroom>(b =>
{

View file

@ -13,7 +13,7 @@ using Volo.Abp.EntityFrameworkCore;
namespace Sozsoft.Platform.Migrations
{
[DbContext(typeof(PlatformDbContext))]
[Migration("20260610203148_Initial")]
[Migration("20260612183658_Initial")]
partial class Initial
{
/// <inheritdoc />
@ -3610,6 +3610,164 @@ namespace Sozsoft.Platform.Migrations
b.ToTable("Sas_H_MenuGroup", (string)null);
});
modelBuilder.Entity("Sozsoft.Platform.Entities.MessengerConversation", b =>
{
b.Property<Guid>("Id")
.HasColumnType("uniqueidentifier");
b.Property<DateTime>("CreationTime")
.HasColumnType("datetime2")
.HasColumnName("CreationTime");
b.Property<Guid?>("CreatorId")
.HasColumnType("uniqueidentifier")
.HasColumnName("CreatorId");
b.Property<Guid?>("DeleterId")
.HasColumnType("uniqueidentifier")
.HasColumnName("DeleterId");
b.Property<DateTime?>("DeletionTime")
.HasColumnType("datetime2")
.HasColumnName("DeletionTime");
b.Property<bool>("IsDeleted")
.ValueGeneratedOnAdd()
.HasColumnType("bit")
.HasDefaultValue(false)
.HasColumnName("IsDeleted");
b.Property<bool>("IsGroup")
.HasColumnType("bit");
b.Property<string>("LastMessagePreview")
.HasMaxLength(512)
.HasColumnType("nvarchar(512)");
b.Property<DateTime?>("LastMessageTime")
.HasColumnType("datetime2");
b.Property<DateTime?>("LastModificationTime")
.HasColumnType("datetime2")
.HasColumnName("LastModificationTime");
b.Property<Guid?>("LastModifierId")
.HasColumnType("uniqueidentifier")
.HasColumnName("LastModifierId");
b.Property<Guid?>("LastSenderId")
.HasColumnType("uniqueidentifier");
b.Property<int>("MessageCount")
.ValueGeneratedOnAdd()
.HasColumnType("int")
.HasDefaultValue(0);
b.Property<string>("ParticipantIdsJson")
.IsRequired()
.HasColumnType("text");
b.Property<string>("ParticipantKey")
.IsRequired()
.HasMaxLength(512)
.HasColumnType("nvarchar(512)");
b.Property<Guid?>("TenantId")
.HasColumnType("uniqueidentifier")
.HasColumnName("TenantId");
b.Property<string>("Title")
.HasMaxLength(256)
.HasColumnType("nvarchar(256)");
b.HasKey("Id");
b.HasIndex("TenantId", "ParticipantKey")
.IsUnique()
.HasFilter("[IsDeleted] = 0");
b.ToTable("Adm_T_MessengerConversation", (string)null);
});
modelBuilder.Entity("Sozsoft.Platform.Entities.MessengerConversationMessage", b =>
{
b.Property<Guid>("Id")
.HasColumnType("uniqueidentifier");
b.Property<string>("AttachmentsJson")
.IsRequired()
.HasColumnType("text");
b.Property<Guid>("ConversationId")
.HasColumnType("uniqueidentifier");
b.Property<DateTime>("CreationTime")
.HasColumnType("datetime2")
.HasColumnName("CreationTime");
b.Property<Guid?>("CreatorId")
.HasColumnType("uniqueidentifier")
.HasColumnName("CreatorId");
b.Property<Guid?>("DeleterId")
.HasColumnType("uniqueidentifier")
.HasColumnName("DeleterId");
b.Property<DateTime?>("DeletionTime")
.HasColumnType("datetime2")
.HasColumnName("DeletionTime");
b.Property<bool>("IsDeleted")
.ValueGeneratedOnAdd()
.HasColumnType("bit")
.HasDefaultValue(false)
.HasColumnName("IsDeleted");
b.Property<DateTime?>("LastModificationTime")
.HasColumnType("datetime2")
.HasColumnName("LastModificationTime");
b.Property<Guid?>("LastModifierId")
.HasColumnType("uniqueidentifier")
.HasColumnName("LastModifierId");
b.Property<string>("RecipientIdsJson")
.IsRequired()
.HasColumnType("text");
b.Property<Guid>("SenderId")
.HasColumnType("uniqueidentifier");
b.Property<string>("SenderName")
.IsRequired()
.HasMaxLength(256)
.HasColumnType("nvarchar(256)");
b.Property<string>("SenderUserName")
.IsRequired()
.HasMaxLength(256)
.HasColumnType("nvarchar(256)");
b.Property<DateTime>("SentAt")
.HasColumnType("datetime2");
b.Property<Guid?>("TenantId")
.HasColumnType("uniqueidentifier")
.HasColumnName("TenantId");
b.Property<string>("Text")
.HasMaxLength(4096)
.HasColumnType("nvarchar(max)");
b.HasKey("Id");
b.HasIndex("ConversationId");
b.HasIndex("TenantId", "ConversationId", "SentAt");
b.ToTable("Adm_T_MessengerConversationMessage", (string)null);
});
modelBuilder.Entity("Sozsoft.Platform.Entities.Note", b =>
{
b.Property<Guid>("Id")
@ -8262,6 +8420,17 @@ namespace Sozsoft.Platform.Migrations
.IsRequired();
});
modelBuilder.Entity("Sozsoft.Platform.Entities.MessengerConversationMessage", b =>
{
b.HasOne("Sozsoft.Platform.Entities.MessengerConversation", "Conversation")
.WithMany("Messages")
.HasForeignKey("ConversationId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.Navigation("Conversation");
});
modelBuilder.Entity("Sozsoft.Platform.Entities.OrderItem", b =>
{
b.HasOne("Sozsoft.Platform.Entities.Order", "Order")
@ -8692,6 +8861,11 @@ namespace Sozsoft.Platform.Migrations
b.Navigation("Events");
});
modelBuilder.Entity("Sozsoft.Platform.Entities.MessengerConversation", b =>
{
b.Navigation("Messages");
});
modelBuilder.Entity("Sozsoft.Platform.Entities.Order", b =>
{
b.Navigation("Items");

View file

@ -594,6 +594,33 @@ namespace Sozsoft.Platform.Migrations
table.PrimaryKey("PK_Adm_T_IpRestriction", x => x.Id);
});
migrationBuilder.CreateTable(
name: "Adm_T_MessengerConversation",
columns: table => new
{
Id = table.Column<Guid>(type: "uniqueidentifier", nullable: false),
TenantId = table.Column<Guid>(type: "uniqueidentifier", nullable: true),
Title = table.Column<string>(type: "nvarchar(256)", maxLength: 256, nullable: true),
ParticipantKey = table.Column<string>(type: "nvarchar(512)", maxLength: 512, nullable: false),
ParticipantIdsJson = table.Column<string>(type: "text", nullable: false),
IsGroup = table.Column<bool>(type: "bit", nullable: false),
LastSenderId = table.Column<Guid>(type: "uniqueidentifier", nullable: true),
LastMessagePreview = table.Column<string>(type: "nvarchar(512)", maxLength: 512, nullable: true),
LastMessageTime = table.Column<DateTime>(type: "datetime2", nullable: true),
MessageCount = table.Column<int>(type: "int", nullable: false, defaultValue: 0),
CreationTime = table.Column<DateTime>(type: "datetime2", nullable: false),
CreatorId = table.Column<Guid>(type: "uniqueidentifier", nullable: true),
LastModificationTime = table.Column<DateTime>(type: "datetime2", nullable: true),
LastModifierId = table.Column<Guid>(type: "uniqueidentifier", nullable: true),
IsDeleted = table.Column<bool>(type: "bit", nullable: false, defaultValue: false),
DeleterId = table.Column<Guid>(type: "uniqueidentifier", nullable: true),
DeletionTime = table.Column<DateTime>(type: "datetime2", nullable: true)
},
constraints: table =>
{
table.PrimaryKey("PK_Adm_T_MessengerConversation", x => x.Id);
});
migrationBuilder.CreateTable(
name: "Adm_T_Note",
columns: table => new
@ -2118,6 +2145,39 @@ namespace Sozsoft.Platform.Migrations
onDelete: ReferentialAction.Restrict);
});
migrationBuilder.CreateTable(
name: "Adm_T_MessengerConversationMessage",
columns: table => new
{
Id = table.Column<Guid>(type: "uniqueidentifier", nullable: false),
TenantId = table.Column<Guid>(type: "uniqueidentifier", nullable: true),
ConversationId = table.Column<Guid>(type: "uniqueidentifier", nullable: false),
SenderId = table.Column<Guid>(type: "uniqueidentifier", nullable: false),
SenderUserName = table.Column<string>(type: "nvarchar(256)", maxLength: 256, nullable: false),
SenderName = table.Column<string>(type: "nvarchar(256)", maxLength: 256, nullable: false),
RecipientIdsJson = table.Column<string>(type: "text", nullable: false),
Text = table.Column<string>(type: "nvarchar(max)", maxLength: 4096, nullable: true),
AttachmentsJson = table.Column<string>(type: "text", nullable: false),
SentAt = table.Column<DateTime>(type: "datetime2", nullable: false),
CreationTime = table.Column<DateTime>(type: "datetime2", nullable: false),
CreatorId = table.Column<Guid>(type: "uniqueidentifier", nullable: true),
LastModificationTime = table.Column<DateTime>(type: "datetime2", nullable: true),
LastModifierId = table.Column<Guid>(type: "uniqueidentifier", nullable: true),
IsDeleted = table.Column<bool>(type: "bit", nullable: false, defaultValue: false),
DeleterId = table.Column<Guid>(type: "uniqueidentifier", nullable: true),
DeletionTime = table.Column<DateTime>(type: "datetime2", nullable: true)
},
constraints: table =>
{
table.PrimaryKey("PK_Adm_T_MessengerConversationMessage", x => x.Id);
table.ForeignKey(
name: "FK_Adm_T_MessengerConversationMessage_Adm_T_MessengerConversation_ConversationId",
column: x => x.ConversationId,
principalTable: "Adm_T_MessengerConversation",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
});
migrationBuilder.CreateTable(
name: "Adm_T_ReportTemplate",
columns: table => new
@ -3570,6 +3630,23 @@ namespace Sozsoft.Platform.Migrations
unique: true,
filter: "[IsDeleted] = 0");
migrationBuilder.CreateIndex(
name: "IX_Adm_T_MessengerConversation_TenantId_ParticipantKey",
table: "Adm_T_MessengerConversation",
columns: new[] { "TenantId", "ParticipantKey" },
unique: true,
filter: "[IsDeleted] = 0");
migrationBuilder.CreateIndex(
name: "IX_Adm_T_MessengerConversationMessage_ConversationId",
table: "Adm_T_MessengerConversationMessage",
column: "ConversationId");
migrationBuilder.CreateIndex(
name: "IX_Adm_T_MessengerConversationMessage_TenantId_ConversationId_SentAt",
table: "Adm_T_MessengerConversationMessage",
columns: new[] { "TenantId", "ConversationId", "SentAt" });
migrationBuilder.CreateIndex(
name: "IX_Adm_T_ReportCategory_TenantId_Name",
table: "Adm_T_ReportCategory",
@ -4148,6 +4225,9 @@ namespace Sozsoft.Platform.Migrations
migrationBuilder.DropTable(
name: "Adm_T_JobPosition");
migrationBuilder.DropTable(
name: "Adm_T_MessengerConversationMessage");
migrationBuilder.DropTable(
name: "Adm_T_Note");
@ -4334,6 +4414,9 @@ namespace Sozsoft.Platform.Migrations
migrationBuilder.DropTable(
name: "Adm_T_Department");
migrationBuilder.DropTable(
name: "Adm_T_MessengerConversation");
migrationBuilder.DropTable(
name: "Adm_T_ReportCategory");

View file

@ -3607,6 +3607,164 @@ namespace Sozsoft.Platform.Migrations
b.ToTable("Sas_H_MenuGroup", (string)null);
});
modelBuilder.Entity("Sozsoft.Platform.Entities.MessengerConversation", b =>
{
b.Property<Guid>("Id")
.HasColumnType("uniqueidentifier");
b.Property<DateTime>("CreationTime")
.HasColumnType("datetime2")
.HasColumnName("CreationTime");
b.Property<Guid?>("CreatorId")
.HasColumnType("uniqueidentifier")
.HasColumnName("CreatorId");
b.Property<Guid?>("DeleterId")
.HasColumnType("uniqueidentifier")
.HasColumnName("DeleterId");
b.Property<DateTime?>("DeletionTime")
.HasColumnType("datetime2")
.HasColumnName("DeletionTime");
b.Property<bool>("IsDeleted")
.ValueGeneratedOnAdd()
.HasColumnType("bit")
.HasDefaultValue(false)
.HasColumnName("IsDeleted");
b.Property<bool>("IsGroup")
.HasColumnType("bit");
b.Property<string>("LastMessagePreview")
.HasMaxLength(512)
.HasColumnType("nvarchar(512)");
b.Property<DateTime?>("LastMessageTime")
.HasColumnType("datetime2");
b.Property<DateTime?>("LastModificationTime")
.HasColumnType("datetime2")
.HasColumnName("LastModificationTime");
b.Property<Guid?>("LastModifierId")
.HasColumnType("uniqueidentifier")
.HasColumnName("LastModifierId");
b.Property<Guid?>("LastSenderId")
.HasColumnType("uniqueidentifier");
b.Property<int>("MessageCount")
.ValueGeneratedOnAdd()
.HasColumnType("int")
.HasDefaultValue(0);
b.Property<string>("ParticipantIdsJson")
.IsRequired()
.HasColumnType("text");
b.Property<string>("ParticipantKey")
.IsRequired()
.HasMaxLength(512)
.HasColumnType("nvarchar(512)");
b.Property<Guid?>("TenantId")
.HasColumnType("uniqueidentifier")
.HasColumnName("TenantId");
b.Property<string>("Title")
.HasMaxLength(256)
.HasColumnType("nvarchar(256)");
b.HasKey("Id");
b.HasIndex("TenantId", "ParticipantKey")
.IsUnique()
.HasFilter("[IsDeleted] = 0");
b.ToTable("Adm_T_MessengerConversation", (string)null);
});
modelBuilder.Entity("Sozsoft.Platform.Entities.MessengerConversationMessage", b =>
{
b.Property<Guid>("Id")
.HasColumnType("uniqueidentifier");
b.Property<string>("AttachmentsJson")
.IsRequired()
.HasColumnType("text");
b.Property<Guid>("ConversationId")
.HasColumnType("uniqueidentifier");
b.Property<DateTime>("CreationTime")
.HasColumnType("datetime2")
.HasColumnName("CreationTime");
b.Property<Guid?>("CreatorId")
.HasColumnType("uniqueidentifier")
.HasColumnName("CreatorId");
b.Property<Guid?>("DeleterId")
.HasColumnType("uniqueidentifier")
.HasColumnName("DeleterId");
b.Property<DateTime?>("DeletionTime")
.HasColumnType("datetime2")
.HasColumnName("DeletionTime");
b.Property<bool>("IsDeleted")
.ValueGeneratedOnAdd()
.HasColumnType("bit")
.HasDefaultValue(false)
.HasColumnName("IsDeleted");
b.Property<DateTime?>("LastModificationTime")
.HasColumnType("datetime2")
.HasColumnName("LastModificationTime");
b.Property<Guid?>("LastModifierId")
.HasColumnType("uniqueidentifier")
.HasColumnName("LastModifierId");
b.Property<string>("RecipientIdsJson")
.IsRequired()
.HasColumnType("text");
b.Property<Guid>("SenderId")
.HasColumnType("uniqueidentifier");
b.Property<string>("SenderName")
.IsRequired()
.HasMaxLength(256)
.HasColumnType("nvarchar(256)");
b.Property<string>("SenderUserName")
.IsRequired()
.HasMaxLength(256)
.HasColumnType("nvarchar(256)");
b.Property<DateTime>("SentAt")
.HasColumnType("datetime2");
b.Property<Guid?>("TenantId")
.HasColumnType("uniqueidentifier")
.HasColumnName("TenantId");
b.Property<string>("Text")
.HasMaxLength(4096)
.HasColumnType("nvarchar(max)");
b.HasKey("Id");
b.HasIndex("ConversationId");
b.HasIndex("TenantId", "ConversationId", "SentAt");
b.ToTable("Adm_T_MessengerConversationMessage", (string)null);
});
modelBuilder.Entity("Sozsoft.Platform.Entities.Note", b =>
{
b.Property<Guid>("Id")
@ -8259,6 +8417,17 @@ namespace Sozsoft.Platform.Migrations
.IsRequired();
});
modelBuilder.Entity("Sozsoft.Platform.Entities.MessengerConversationMessage", b =>
{
b.HasOne("Sozsoft.Platform.Entities.MessengerConversation", "Conversation")
.WithMany("Messages")
.HasForeignKey("ConversationId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.Navigation("Conversation");
});
modelBuilder.Entity("Sozsoft.Platform.Entities.OrderItem", b =>
{
b.HasOne("Sozsoft.Platform.Entities.Order", "Order")
@ -8689,6 +8858,11 @@ namespace Sozsoft.Platform.Migrations
b.Navigation("Events");
});
modelBuilder.Entity("Sozsoft.Platform.Entities.MessengerConversation", b =>
{
b.Navigation("Messages");
});
modelBuilder.Entity("Sozsoft.Platform.Entities.Order", b =>
{
b.Navigation("Items");

Binary file not shown.

Before

Width:  |  Height:  |  Size: 3.8 KiB

After

Width:  |  Height:  |  Size: 12 KiB

View file

@ -1,5 +1,5 @@
{
"commit": "6098758",
"commit": "7632c9e",
"releases": [
{
"version": "1.1.04",

View file

@ -7,16 +7,27 @@ import { toast } from '@/components/ui'
import { AVATAR_URL } from '@/constants/app.constant'
import {
getMessengerContacts,
getMessengerMessages,
MessengerAttachmentDto,
MessengerContactDto,
MessengerMessageDto,
uploadMessengerAttachment,
} from '@/services/messenger.service'
import { messengerSignalR, MessengerMessageDto } from '@/services/messenger.signalr'
import { messengerSignalR } from '@/services/messenger.signalr'
import { useStoreState } from '@/store'
import dayjs from 'dayjs'
import EmojiPicker, { EmojiClickData } from 'emoji-picker-react'
import { useEffect, useMemo, useRef, useState } from 'react'
import { FaArrowLeft, FaPaperclip, FaRegSmile, FaSearch, FaTimes, FaTrash } from 'react-icons/fa'
import {
FaArrowLeft,
FaCompress,
FaExpand,
FaPaperclip,
FaRegSmile,
FaSearch,
FaTimes,
FaTrash,
} from 'react-icons/fa'
import { IoCheckmarkDone, IoChatbubbleEllipsesOutline, IoSend } from 'react-icons/io5'
const MAX_UPLOAD_SIZE = 10 * 1024 * 1024
@ -26,10 +37,14 @@ const formatFileSize = (size: number) => {
return `${(size / (1024 * 1024)).toFixed(1)} MB`
}
const canDeleteMessage = (message: MessengerMessageDto, userId: string) =>
message.senderId === userId && dayjs().diff(dayjs(message.sentAt), 'minute', true) < 10
const MessengerWidget = () => {
const auth = useStoreState((state) => state.auth)
const tenantId = auth.user.tenantId || auth.tenant?.tenantId
const fileInputRef = useRef<HTMLInputElement>(null)
const messageInputRef = useRef<HTMLTextAreaElement>(null)
const messagesEndRef = useRef<HTMLDivElement>(null)
const [open, setOpen] = useState(false)
@ -43,25 +58,38 @@ const MessengerWidget = () => {
const [uploading, setUploading] = useState(false)
const [showEmoji, setShowEmoji] = useState(false)
const [multiSelect, setMultiSelect] = useState(false)
const [unread, setUnread] = useState(0)
const [maximized, setMaximized] = useState(false)
const [unreadByContact, setUnreadByContact] = useState<Record<string, number>>({})
const unread = useMemo(
() => Object.values(unreadByContact).reduce((total, count) => total + count, 0),
[unreadByContact],
)
useEffect(() => {
if (!auth.session.signedIn) return
const unsubscribeMessage = messengerSignalR.onMessage((message) => {
setMessages((prev) => (prev.some((item) => item.id === message.id) ? prev : [...prev, message]))
if (!open && message.senderId !== auth.user.id) {
setUnread((count) => count + 1)
if (message.senderId !== auth.user.id && (!open || !selectedIds.includes(message.senderId))) {
setUnreadByContact((current) => ({
...current,
[message.senderId]: (current[message.senderId] || 0) + 1,
}))
}
})
const unsubscribeMessageDeleted = messengerSignalR.onMessageDeleted((message) => {
setMessages((prev) => prev.filter((item) => item.id !== message.messageId))
})
const unsubscribeState = messengerSignalR.onStateChange(setConnected)
messengerSignalR.start()
return () => {
unsubscribeMessage()
unsubscribeMessageDeleted()
unsubscribeState()
}
}, [auth.session.signedIn, auth.user.id, open])
}, [auth.session.signedIn, auth.user.id, open, selectedIds])
useEffect(() => {
if (!auth.session.signedIn) {
@ -71,7 +99,6 @@ const MessengerWidget = () => {
useEffect(() => {
if (!open) return
setUnread(0)
getMessengerContacts(filter)
.then((response) => setContacts(response.data))
.catch(() => setContacts([]))
@ -81,13 +108,35 @@ const MessengerWidget = () => {
messagesEndRef.current?.scrollIntoView({ behavior: 'smooth' })
}, [messages, open])
useEffect(() => {
if (!open || selectedIds.length === 0) return
const timer = window.setTimeout(() => {
messageInputRef.current?.focus()
}, 0)
return () => window.clearTimeout(timer)
}, [open, selectedIds])
useEffect(() => {
if (!open || selectedIds.length === 0) return
getMessengerMessages({
participantIds: selectedIds,
maxResultCount: 50,
sorting: 'sentAt desc',
})
.then((response) => setMessages(response.data))
.catch(() => setMessages([]))
}, [open, selectedIds])
const selectedContacts = useMemo(
() => contacts.filter((contact) => selectedIds.includes(contact.id)),
[contacts, selectedIds],
)
const visibleMessages = useMemo(() => {
if (selectedIds.length === 0) return messages.slice(-30)
if (selectedIds.length === 0) return []
return messages.filter((message) => {
const isOwn = message.senderId === auth.user.id
@ -97,6 +146,14 @@ const MessengerWidget = () => {
}, [auth.user.id, messages, selectedIds])
const toggleContact = (id: string) => {
setUnreadByContact((current) => {
if (!current[id]) return current
const next = { ...current }
delete next[id]
return next
})
if (!multiSelect) {
setSelectedIds([id])
return
@ -114,6 +171,16 @@ const MessengerWidget = () => {
})
}
const closeWidget = () => {
setOpen(false)
setMaximized(false)
setSelectedIds([])
setMessages([])
setText('')
setAttachments([])
setShowEmoji(false)
}
const handleFiles = async (files: FileList | null) => {
if (!files?.length) return
@ -153,21 +220,38 @@ const MessengerWidget = () => {
setText('')
setAttachments([])
setShowEmoji(false)
window.setTimeout(() => {
messageInputRef.current?.focus()
}, 0)
} catch {
toast.push(<Notification title="Mesaj gönderilemedi" type="danger" />, { placement: 'top-end' })
}
}
const deleteMessage = async (messageId: string) => {
if (!window.confirm('Mesaj silinsin mi?')) return
try {
await messengerSignalR.deleteMessage(messageId)
} catch {
toast.push(<Notification title="Mesaj silinemedi" type="danger" />, { placement: 'top-end' })
}
}
const onEmojiClick = (emoji: EmojiClickData) => {
setText((current) => `${current}${emoji.emoji}`)
}
if (!auth.session.signedIn) return null
const panelClassName = maximized
? 'pointer-events-auto fixed inset-2 overflow-hidden rounded-lg border border-gray-200 bg-white shadow-2xl dark:border-gray-700 dark:bg-gray-800 sm:inset-4'
: 'pointer-events-auto fixed inset-x-2 bottom-20 top-3 overflow-hidden rounded-lg border border-gray-200 bg-white shadow-2xl dark:border-gray-700 dark:bg-gray-800 sm:absolute sm:inset-auto sm:bottom-16 sm:right-0 sm:h-[min(70vh,640px)] sm:w-[min(78vw,800px)]'
return (
<div className="pointer-events-none fixed bottom-5 right-5 z-50 h-14 w-14">
{open && (
<div className="pointer-events-auto fixed inset-x-2 bottom-20 top-3 overflow-hidden rounded-lg border border-gray-200 bg-white shadow-2xl dark:border-gray-700 dark:bg-gray-800 sm:absolute sm:inset-auto sm:bottom-16 sm:right-0 sm:h-[min(70vh,640px)] sm:w-[min(78vw,800px)]">
<div className={panelClassName}>
<div className="flex h-full min-h-0">
<aside
className={`min-h-0 w-full shrink-0 flex-col border-gray-200 bg-gray-50 dark:border-gray-700 dark:bg-gray-900 sm:flex sm:w-72 sm:border-r ${
@ -187,7 +271,17 @@ const MessengerWidget = () => {
</div>
</div>
</div>
<Button size="xs" variant="plain" icon={<FaTimes />} onClick={() => setOpen(false)} />
<div className="flex shrink-0 items-center gap-1 sm:hidden">
<Tooltip title={maximized ? 'Küçült' : 'Büyüt'}>
<Button
size="xs"
variant="plain"
icon={maximized ? <FaCompress /> : <FaExpand />}
onClick={() => setMaximized((value) => !value)}
/>
</Tooltip>
<Button size="xs" variant="plain" icon={<FaTimes />} onClick={closeWidget} />
</div>
</div>
<div className="space-y-2 p-3">
@ -211,6 +305,7 @@ const MessengerWidget = () => {
<div className="min-h-0 flex-1 overflow-y-auto px-2 pb-2">
{contacts.map((contact) => {
const active = selectedIds.includes(contact.id)
const contactUnread = unreadByContact[contact.id] || 0
return (
<button
key={contact.id}
@ -227,7 +322,12 @@ const MessengerWidget = () => {
<span className="block truncate text-sm font-medium">{contact.fullName}</span>
<span className="block truncate text-xs text-gray-500">{contact.email || contact.userName}</span>
</span>
{active && <IoCheckmarkDone className="shrink-0 text-lg" />}
{contactUnread > 0 && (
<span className="shrink-0 rounded-full bg-red-500 px-2 py-0.5 text-xs font-semibold text-white">
{contactUnread > 99 ? '99+' : contactUnread}
</span>
)}
{active && contactUnread === 0 && <IoCheckmarkDone className="shrink-0 text-lg" />}
</button>
)
})}
@ -277,6 +377,17 @@ const MessengerWidget = () => {
</div>
</div>
</div>
<div className="flex shrink-0 items-center gap-1">
<Tooltip title={maximized ? 'Küçült' : 'Büyüt'}>
<Button
size="xs"
variant="plain"
icon={maximized ? <FaCompress /> : <FaExpand />}
onClick={() => setMaximized((value) => !value)}
/>
</Tooltip>
<Button size="xs" variant="plain" icon={<FaTimes />} onClick={closeWidget} />
</div>
</div>
<div className="flex-1 overflow-y-auto bg-white px-4 py-3 dark:bg-gray-800">
@ -288,16 +399,31 @@ const MessengerWidget = () => {
{visibleMessages.map((message) => {
const own = message.senderId === auth.user.id
const deletable = canDeleteMessage(message, auth.user.id)
return (
<div key={message.id} className={`mb-3 flex ${own ? 'justify-end' : 'justify-start'}`}>
<div className={`group flex max-w-[76%] items-start gap-1.5 ${own ? 'flex-row-reverse' : ''}`}>
<div
className={`max-w-[76%] rounded-lg px-3 py-2 shadow-sm ${
className={`rounded-lg px-3 py-2 shadow-sm ${
own
? 'bg-emerald-600 text-white'
: 'bg-gray-100 text-gray-900 dark:bg-gray-700 dark:text-gray-100'
}`}
>
{!own && <div className="mb-1 text-xs font-semibold">{message.senderName}</div>}
<div className="mb-1 flex items-center gap-1.5 text-xs font-semibold">
<Avatar
size={20}
shape="circle"
src={
own
? auth.user.avatar || AVATAR_URL(auth.user.id, tenantId)
: AVATAR_URL(message.senderId, message.tenantId || tenantId)
}
/>
<span className="max-w-[160px] truncate">
{own ? auth.user.name || auth.user.userName || 'Ben' : message.senderName}
</span>
</div>
{message.text && <div className="whitespace-pre-wrap text-sm leading-5">{message.text}</div>}
{message.attachments.length > 0 && (
<div className="mt-2 space-y-1">
@ -317,10 +443,25 @@ const MessengerWidget = () => {
))}
</div>
)}
<div className={`mt-1 text-[11px] ${own ? 'text-emerald-100' : 'text-gray-500'}`}>
{dayjs(message.sentAt).format('HH:mm')}
<div
className={`mt-1 flex items-center gap-2 text-[11px] ${
own ? 'justify-end text-emerald-100' : 'justify-start text-gray-500'
}`}
>
<span>{dayjs(message.sentAt).format('HH:mm')}</span>
</div>
</div>
{own && deletable && (
<button
type="button"
title="Mesaji sil"
className="mt-1 flex h-5 w-5 shrink-0 items-center justify-center rounded-full text-[10px] text-gray-400 opacity-0 transition hover:bg-red-500 hover:text-white group-hover:opacity-100 focus:opacity-100"
onClick={() => deleteMessage(message.id)}
>
<FaTrash />
</button>
)}
</div>
</div>
)
})}
@ -391,6 +532,7 @@ const MessengerWidget = () => {
/>
</Tooltip>
<textarea
ref={messageInputRef}
value={text}
placeholder="Mesaj yazın"
rows={1}
@ -424,12 +566,12 @@ const MessengerWidget = () => {
<Tooltip title="Messenger">
<button
type="button"
onClick={() => setOpen((value) => !value)}
onClick={() => (open ? closeWidget() : setOpen(true))}
className="pointer-events-auto relative flex h-12 w-12 items-center justify-center rounded-full bg-emerald-600 text-2xl text-white shadow-xl transition hover:bg-emerald-700"
>
<IoChatbubbleEllipsesOutline />
{unread > 0 && (
<span className="absolute min-w-[20px] rounded-full bg-red-500 px-1.5 py-0.5 text-xs font-semibold text-white">
<span className="absolute -right-1 -top-1 flex min-h-[20px] min-w-[20px] items-center justify-center rounded-full bg-red-500 px-1.5 text-xs font-semibold leading-none text-white ring-2 ring-white dark:ring-gray-800">
{unread > 99 ? '99+' : unread}
</span>
)}

View file

@ -19,6 +19,34 @@ export interface MessengerAttachmentDto {
url: string
}
export interface MessengerGetMessagesInput {
conversationId?: string
participantIds: string[]
skipCount?: number
maxResultCount?: number
sorting?: string
}
export interface MessengerMessageDto {
id: string
tenantId?: string
conversationId: string
senderId: string
senderUserName: string
senderName: string
recipientIds: string[]
text?: string
attachments: MessengerAttachmentDto[]
sentAt: string
}
export interface MessengerMessageDeletedDto {
messageId: string
conversationId: string
senderId: string
recipientIds: string[]
}
export const getMessengerContacts = (filter?: string) =>
apiService.fetchData<MessengerContactDto[]>({
method: 'GET',
@ -26,6 +54,13 @@ export const getMessengerContacts = (filter?: string) =>
params: filter ? { filter } : undefined,
})
export const getMessengerMessages = (data: MessengerGetMessagesInput) =>
apiService.fetchData<MessengerMessageDto[]>({
method: 'POST',
url: '/api/app/messenger/messages',
data,
})
export const uploadMessengerAttachment = (file: File) => {
const formData = new FormData()
formData.append('file', file)

View file

@ -1,31 +1,22 @@
import { store } from '@/store/store'
import * as signalR from '@microsoft/signalr'
import { MessengerAttachmentDto } from './messenger.service'
export interface MessengerMessageDto {
id: string
tenantId?: string
senderId: string
senderUserName: string
senderName: string
recipientIds: string[]
text?: string
attachments: MessengerAttachmentDto[]
sentAt: string
}
import { MessengerAttachmentDto, MessengerMessageDeletedDto, MessengerMessageDto } from './messenger.service'
export interface MessengerSendMessageDto {
conversationId?: string
recipientIds: string[]
text?: string
attachments: MessengerAttachmentDto[]
}
type MessageHandler = (message: MessengerMessageDto) => void
type MessageDeletedHandler = (message: MessengerMessageDeletedDto) => void
type StateHandler = (connected: boolean) => void
class MessengerSignalRService {
private connection?: signalR.HubConnection
private messageHandlers = new Set<MessageHandler>()
private messageDeletedHandlers = new Set<MessageDeletedHandler>()
private stateHandlers = new Set<StateHandler>()
private createConnection() {
@ -43,6 +34,10 @@ class MessengerSignalRService {
this.messageHandlers.forEach((handler) => handler(message))
})
this.connection.on('MessengerMessageDeleted', (message: MessengerMessageDeletedDto) => {
this.messageDeletedHandlers.forEach((handler) => handler(message))
})
this.connection.onreconnected(() => this.emitState(true))
this.connection.onreconnecting(() => this.emitState(false))
this.connection.onclose(() => this.emitState(false))
@ -87,11 +82,25 @@ class MessengerSignalRService {
await this.connection.invoke('SendMessage', input)
}
async deleteMessage(messageId: string) {
await this.start()
if (this.connection?.state !== signalR.HubConnectionState.Connected) {
throw new Error('Messenger bağlantısı yok')
}
await this.connection.invoke('DeleteMessage', messageId)
}
onMessage(handler: MessageHandler) {
this.messageHandlers.add(handler)
return () => this.messageHandlers.delete(handler)
}
onMessageDeleted(handler: MessageDeletedHandler) {
this.messageDeletedHandlers.add(handler)
return () => this.messageDeletedHandlers.delete(handler)
}
onStateChange(handler: StateHandler) {
this.stateHandlers.add(handler)
handler(this.getConnectionState())