using Microsoft.EntityFrameworkCore; using NetCoreServer; using Server.DB; using Server.DB.Documents; using Server.Services; using Wonderking.Game.Data.Character; using Wonderking.Game.Mapping; using Wonderking.Packets.Incoming; using Wonderking.Packets.Outgoing; using Wonderking.Packets.Outgoing.Data; namespace Server.PacketHandlers; public class CharacterCreationHandler : IPacketHandler { private readonly CharacterStatsMappingConfiguration _characterStatsMapping; private readonly ItemObjectPoolService _itemObjectPoolService; private readonly WonderkingContext _wonderkingContext; public CharacterCreationHandler(WonderkingContext wonderkingContext, ItemObjectPoolService itemObjectPoolService, CharacterStatsMappingConfiguration characterStatsMappingConfiguration) { _wonderkingContext = wonderkingContext; _itemObjectPoolService = itemObjectPoolService; _characterStatsMapping = characterStatsMappingConfiguration; } public async Task HandleAsync(CharacterCreationPacket packet, TcpSession session) { var authSession = session as AuthSession; if (authSession is null) { return; } var account = _wonderkingContext.Accounts.FirstOrDefault(a => a.Id == authSession.AccountId); var firstJobConfig = SelectFirstJobConfig(packet.FirstJob); var items = CreateDefaultItems(packet, firstJobConfig); var toBeAddedCharacter = CreateDefaultCharacter(packet, account, items, firstJobConfig); account?.Characters.Add(toBeAddedCharacter); await _wonderkingContext.SaveChangesAsync().ConfigureAwait(true); var character = await _wonderkingContext.Characters.AsNoTrackingWithIdentityResolution().AsSplitQuery() .Where(c => c.Account.Id == authSession.AccountId && c.Name == packet.Name) .Select(c => new CharacterData { Name = c.Name, Job = c.JobData, Gender = c.Gender, Level = c.Level, Experience = c.Experience, Stats = c.BaseStats, Health = c.Health, Mana = c.Mana, EquippedItems = GetItemIDsByInventoryTab(c.InventoryItems .Where(item => item.InventoryTab == InventoryTab.WornEquipment) .Select(item => new Tuple(item.ItemId, item.Slot)).AsEnumerable()), EquippedCashItems = GetItemIDsByInventoryTab(c.InventoryItems .Where(item => item.InventoryTab == InventoryTab.WornCashEquipment) .Select(item => new Tuple(item.ItemId, item.Slot)).AsEnumerable()) }).FirstAsync().ConfigureAwait(true); await authSession.SendAsync(new CharacterCreationResponsePacket { Character = character, Slot = packet.Slot, isDuplicate = false }).ConfigureAwait(false); } private InventoryItem[] CreateDefaultItems(CharacterCreationPacket packet, JobSpecificMapping firstJobConfig) { var mappedDefaultItems = _characterStatsMapping.DefaultCharacterMapping.Items .Select(i => _itemObjectPoolService.GetBaseInventoryItem(i.Id, i.Quantity)).ToArray(); var mappedJobItems = firstJobConfig.Items .Select(i => _itemObjectPoolService.GetBaseInventoryItem(i.Id, i.Quantity)).ToArray(); var defaultItems = CreateChosenItems(packet); InventoryItem[] items = [ .. mappedDefaultItems, .. mappedJobItems, .. defaultItems ]; return items; } private static Character CreateDefaultCharacter(CharacterCreationPacket packet, Account account, InventoryItem[] items, JobSpecificMapping firstJobConfig) { return new Character { Account = account, MapId = 300, Name = packet.Name, LastXCoordinate = 113, LastYCoordinate = 0, PvPLevel = PvPLevel.None, Gender = packet.Gender, Experience = 0, Level = 1, InventoryItems = items, BaseStats = firstJobConfig.BaseStats, JobData = new JobData { FirstJob = packet.FirstJob, SecondJob = 0, ThirdJob = 0, FourthJob = 0 }, Health = CalculateCurrentHealth(1, firstJobConfig), Mana = CalculateCurrentMana(1, firstJobConfig) }; } private JobSpecificMapping SelectFirstJobConfig(byte firstJob) { return firstJob switch { 1 => _characterStatsMapping.Swordsman, 2 => _characterStatsMapping.Mage, 3 => _characterStatsMapping.Thief, 4 => _characterStatsMapping.Scout, _ => _characterStatsMapping.Swordsman }; } private InventoryItem[] CreateChosenItems(CharacterCreationPacket packet) { return new[] { _itemObjectPoolService.GetBaseInventoryItem((ushort)((packet.FirstJob - 1) * 6 + ((byte)packet.Gender - 1) * 3 + packet.Hair + 1)), _itemObjectPoolService.GetBaseInventoryItem((ushort)((packet.FirstJob - 1) * 6 + ((byte)packet.Gender - 1) * 3 + packet.Eyes + 25)), _itemObjectPoolService.GetBaseInventoryItem((ushort)(((byte)packet.Gender - 1) * 3 + packet.Shirt + 49)), _itemObjectPoolService.GetBaseInventoryItem((ushort)(((byte)packet.Gender - 1) * 3 + packet.Pants + 58)) }; } private static int CalculateCurrentHealth(ushort level, JobSpecificMapping firstJobConfig) { return (int)((level - 1) * firstJobConfig.DynamicStats.HealthPerLevel + firstJobConfig.BaseStats.Vitality * firstJobConfig.DynamicStats.HealthPerVitality); } private static int CalculateCurrentMana(ushort level, JobSpecificMapping firstJobConfig) { return (int)((level - 1) * firstJobConfig.DynamicStats.ManaPerLevel + firstJobConfig.BaseStats.Wisdom * firstJobConfig.DynamicStats.ManaPerWisdom); } private static ushort[] GetItemIDsByInventoryTab(IEnumerable> items) { var ids = new ushort[20]; ids.AsSpan().Clear(); foreach (var item in items) { if (item.Item2 > 20) { continue; } ids[item.Item2] = item.Item1; } return ids; } }