增加float 16 类型的支持
This commit is contained in:
parent
cd81d27f38
commit
f970fdff27
98
Server/Models/BattleStats.cs
Normal file
98
Server/Models/BattleStats.cs
Normal file
@ -0,0 +1,98 @@
|
||||
using PSO2SERVER.Protocol.Packets;
|
||||
using PSO2SERVER.Zone;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace PSO2SERVER.Models
|
||||
{
|
||||
// Class to represent PlayerStats
|
||||
public class BattlePlayerStats
|
||||
{
|
||||
public uint MaxHp { get; set; } = 0;
|
||||
public uint Hp { get; set; } = 0;
|
||||
public uint Dex { get; set; } = 0;
|
||||
|
||||
public uint BaseMelPwr { get; set; } = 0;
|
||||
public uint WeaponMelPwr { get; set; } = 0;
|
||||
public uint BaseRngPwr { get; set; } = 0;
|
||||
public uint WeaponRngPwr { get; set; } = 0;
|
||||
public uint BaseTecPwr { get; set; } = 0;
|
||||
public uint WeaponTecPwr { get; set; } = 0;
|
||||
|
||||
public uint BaseMelDef { get; set; } = 0;
|
||||
public uint BaseRngDef { get; set; } = 0;
|
||||
public uint BaseTecDef { get; set; } = 0;
|
||||
|
||||
// Constructor to initialize with default values
|
||||
public BattlePlayerStats()
|
||||
{
|
||||
}
|
||||
}
|
||||
|
||||
// Class to represent EnemyStats
|
||||
public class BattleEnemyStats
|
||||
{
|
||||
public string Name { get; set; } = string.Empty;
|
||||
public uint Level { get; set; } = 0;
|
||||
public uint Exp { get; set; } = 0;
|
||||
public PSOLocation Pos { get; set; } = new PSOLocation();
|
||||
|
||||
public uint MaxHp { get; set; } = 0;
|
||||
public uint Hp { get; set; } = 0;
|
||||
public uint Dex { get; set; } = 0;
|
||||
|
||||
public uint MaxMelPwr { get; set; } = 0;
|
||||
public uint MinMelPwr { get; set; } = 0;
|
||||
public uint MaxRngPwr { get; set; } = 0;
|
||||
public uint MinRngPwr { get; set; } = 0;
|
||||
public uint MaxTecPwr { get; set; } = 0;
|
||||
public uint MinTecPwr { get; set; } = 0;
|
||||
|
||||
public uint MelDef { get; set; } = 0;
|
||||
public uint RngDef { get; set; } = 0;
|
||||
public uint TecDef { get; set; } = 0;
|
||||
|
||||
public List<EnemyHitbox> Hitboxes { get; set; } = new List<EnemyHitbox>();
|
||||
|
||||
// Constructor to initialize with default values
|
||||
public BattleEnemyStats()
|
||||
{
|
||||
}
|
||||
}
|
||||
|
||||
// Enum to represent the BattleResult type
|
||||
public enum BattleResultType
|
||||
{
|
||||
Damaged,
|
||||
Killed
|
||||
}
|
||||
|
||||
// Class to represent the BattleResult
|
||||
public class BattleResult
|
||||
{
|
||||
public BattleResultType ResultType { get; set; }
|
||||
|
||||
public DamageReceivePacket DmgPacket { get; set; }
|
||||
public EnemyKilledPacket KillPacket { get; set; }
|
||||
public uint ExpAmount { get; set; }
|
||||
|
||||
// Constructor for Damaged result
|
||||
public BattleResult(DamageReceivePacket dmgPacket)
|
||||
{
|
||||
ResultType = BattleResultType.Damaged;
|
||||
DmgPacket = dmgPacket;
|
||||
}
|
||||
|
||||
// Constructor for Killed result
|
||||
public BattleResult(DamageReceivePacket dmgPacket, EnemyKilledPacket killPacket, uint expAmount)
|
||||
{
|
||||
ResultType = BattleResultType.Killed;
|
||||
DmgPacket = dmgPacket;
|
||||
KillPacket = killPacket;
|
||||
ExpAmount = expAmount;
|
||||
}
|
||||
}
|
||||
}
|
174
Server/Models/Stats.cs
Normal file
174
Server/Models/Stats.cs
Normal file
@ -0,0 +1,174 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using Newtonsoft.Json;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace PSO2SERVER.Models
|
||||
{
|
||||
public enum AttackType
|
||||
{
|
||||
Mel = 0,
|
||||
Rng = 1,
|
||||
Tec = 2
|
||||
}
|
||||
|
||||
public class LevelStats
|
||||
{
|
||||
public ulong ExpToNext { get; set; } = 0;
|
||||
public float Hp { get; set; } = 0;
|
||||
public float Pp { get; set; } = 0;
|
||||
public float MelPow { get; set; } = 0;
|
||||
public float RngPow { get; set; } = 0;
|
||||
public float TecPow { get; set; } = 0;
|
||||
public float Dex { get; set; } = 0;
|
||||
public float MelDef { get; set; } = 0;
|
||||
public float RngDef { get; set; } = 0;
|
||||
public float TecDef { get; set; } = 0;
|
||||
}
|
||||
|
||||
public class StatMultipliers
|
||||
{
|
||||
public sbyte Hp { get; set; } = 0;
|
||||
public sbyte MelPow { get; set; } = 0;
|
||||
public sbyte RngPow { get; set; } = 0;
|
||||
public sbyte TecPow { get; set; } = 0;
|
||||
public sbyte Dex { get; set; } = 0;
|
||||
public sbyte MelDef { get; set; } = 0;
|
||||
public sbyte RngDef { get; set; } = 0;
|
||||
public sbyte TecDef { get; set; } = 0;
|
||||
}
|
||||
|
||||
public class ClassStatsStored
|
||||
{
|
||||
public ClassType Class { get; set; }
|
||||
public List<LevelStats> Stats { get; set; } = new List<LevelStats>();
|
||||
}
|
||||
|
||||
public class PlayerStats
|
||||
{
|
||||
public List<List<LevelStats>> Stats { get; set; } = new List<List<LevelStats>>();
|
||||
public List<StatMultipliers> Modifiers { get; set; } = new List<StatMultipliers>();
|
||||
}
|
||||
|
||||
public class RaceModifierStored
|
||||
{
|
||||
public StatMultipliers HumanMale { get; set; } = new StatMultipliers();
|
||||
public StatMultipliers HumanFemale { get; set; } = new StatMultipliers();
|
||||
public StatMultipliers NewmanMale { get; set; } = new StatMultipliers();
|
||||
public StatMultipliers NewmanFemale { get; set; } = new StatMultipliers();
|
||||
public StatMultipliers CastMale { get; set; } = new StatMultipliers();
|
||||
public StatMultipliers CastFemale { get; set; } = new StatMultipliers();
|
||||
public StatMultipliers DeumanMale { get; set; } = new StatMultipliers();
|
||||
public StatMultipliers DeumanFemale { get; set; } = new StatMultipliers();
|
||||
}
|
||||
|
||||
public class EnemyLevelBaseStats
|
||||
{
|
||||
public uint Level { get; set; } = 0;
|
||||
public float Exp { get; set; } = 0;
|
||||
public float Hp { get; set; } = 0;
|
||||
public float MaxMelDmg { get; set; } = 0;
|
||||
public float MinMelDmg { get; set; } = 0;
|
||||
public float MaxRngDmg { get; set; } = 0;
|
||||
public float MinRngDmg { get; set; } = 0;
|
||||
public float MaxTecDmg { get; set; } = 0;
|
||||
public float MinTecDmg { get; set; } = 0;
|
||||
public float MelDef { get; set; } = 0;
|
||||
public float RngDef { get; set; } = 0;
|
||||
public float TecDef { get; set; } = 0;
|
||||
public float Dex { get; set; } = 0;
|
||||
}
|
||||
|
||||
public class EnemyHitbox
|
||||
{
|
||||
public string Name { get; set; } = string.Empty;
|
||||
public uint HitboxId { get; set; } = 0;
|
||||
public float DamageMul { get; set; } = 1.0f;
|
||||
public float MelMul { get; set; } = 1.0f;
|
||||
public float RngMul { get; set; } = 1.0f;
|
||||
public float TecMul { get; set; } = 1.0f;
|
||||
public float FireMul { get; set; } = 1.0f;
|
||||
public float IceMul { get; set; } = 1.0f;
|
||||
public float ThunderMul { get; set; } = 1.0f;
|
||||
public float WindMul { get; set; } = 1.0f;
|
||||
public float LightMul { get; set; } = 1.0f;
|
||||
public float DarkMul { get; set; } = 1.0f;
|
||||
}
|
||||
|
||||
public class EnemyBaseStats
|
||||
{
|
||||
public List<EnemyLevelBaseStats> Levels { get; set; } = new List<EnemyLevelBaseStats>();
|
||||
}
|
||||
|
||||
public class EnemyStats
|
||||
{
|
||||
public List<EnemyLevelBaseStats> Levels { get; set; } = new List<EnemyLevelBaseStats>();
|
||||
public List<EnemyHitbox> Hitboxes { get; set; } = new List<EnemyHitbox>();
|
||||
}
|
||||
|
||||
public class NamedEnemyStats
|
||||
{
|
||||
public string Name { get; set; } = string.Empty;
|
||||
public EnemyStats Stats { get; set; } = new EnemyStats();
|
||||
}
|
||||
|
||||
public class AllEnemyStats
|
||||
{
|
||||
public EnemyBaseStats Base { get; set; } = new EnemyBaseStats();
|
||||
public Dictionary<string, EnemyStats> Enemies { get; set; } = new Dictionary<string, EnemyStats>();
|
||||
}
|
||||
|
||||
public class AttackStatsReadable
|
||||
{
|
||||
public string AttackName { get; set; } = string.Empty;
|
||||
public string DamageName { get; set; } = string.Empty;
|
||||
public AttackType AttackType { get; set; } = AttackType.Mel;
|
||||
public AttackType DefenseType { get; set; } = AttackType.Mel;
|
||||
public DamageTypeReadable Damage { get; set; } = new DamageTypeReadable();
|
||||
}
|
||||
|
||||
public class AttackStats
|
||||
{
|
||||
public uint AttackId { get; set; } = 0;
|
||||
public uint DamageId { get; set; } = 0;
|
||||
public AttackType AttackType { get; set; } = AttackType.Mel;
|
||||
public AttackType DefenseType { get; set; } = AttackType.Mel;
|
||||
public DamageType Damage { get; set; } = new DamageType();
|
||||
}
|
||||
|
||||
public class DamageTypeReadable
|
||||
{
|
||||
public static DamageTypeReadable Default => new DamageTypeReadable { Mul = 1.0f };
|
||||
|
||||
public float Mul { get; set; } = 1.0f;
|
||||
public string Name { get; set; } = string.Empty;
|
||||
}
|
||||
|
||||
public class DamageType
|
||||
{
|
||||
public float Mul { get; set; } = 1.0f;
|
||||
public Tuple<uint, float> Pa { get; set; } = new Tuple<uint, float>(0, 1.0f);
|
||||
|
||||
public DamageType() { }
|
||||
|
||||
public DamageType(float mul)
|
||||
{
|
||||
Mul = mul;
|
||||
}
|
||||
|
||||
public DamageType(Tuple<uint, float> pa)
|
||||
{
|
||||
Pa = pa;
|
||||
}
|
||||
}
|
||||
|
||||
public class DamageTypeConverter
|
||||
{
|
||||
public static DamageType ConvertToDamageType(DamageTypeReadable value)
|
||||
{
|
||||
return new DamageType(value.Mul);
|
||||
}
|
||||
}
|
||||
}
|
@ -4,6 +4,7 @@ using System;
|
||||
using System.Collections.Generic;
|
||||
using System.IO;
|
||||
using System.Runtime.InteropServices;
|
||||
using System.Runtime.Remoting.Messaging;
|
||||
using System.Text;
|
||||
|
||||
namespace PSO2SERVER.Protocol
|
||||
@ -143,5 +144,30 @@ namespace PSO2SERVER.Protocol
|
||||
|
||||
return pos;
|
||||
}
|
||||
|
||||
// 读取 Half 类型(16 位浮动点数)
|
||||
public Half ReadHalf()
|
||||
{
|
||||
// 读取 2 个字节
|
||||
byte[] bytes = ReadBytes(2);
|
||||
|
||||
// 使用 System.Half 来转换字节为 Half 类型
|
||||
if (bytes.Length < 2)
|
||||
{
|
||||
throw new EndOfStreamException("Not enough data to read Half value.");
|
||||
}
|
||||
|
||||
// 返回 Half 类型
|
||||
return BitConverter.ToUInt16(bytes, 0).ToHalf();
|
||||
}
|
||||
}
|
||||
|
||||
// Extension method to convert UInt16 to Half
|
||||
public static class HalfExtensions
|
||||
{
|
||||
public static Half ToHalf(this ushort value)
|
||||
{
|
||||
return new Half(value);
|
||||
}
|
||||
}
|
||||
}
|
@ -301,5 +301,11 @@ namespace PSO2SERVER.Protocol
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// 写入 Half 类型(16 位浮动点数)
|
||||
public void Write(Half value)
|
||||
{
|
||||
Write(Half.GetBytes(value)); // 写入 2 个字节
|
||||
}
|
||||
}
|
||||
}
|
@ -8,6 +8,33 @@ namespace PSO2SERVER.Protocol.Packets
|
||||
{
|
||||
public class EnemyKilledPacket : Packet
|
||||
{
|
||||
/// Player that receives this packet.
|
||||
public ObjectHeader Receiver { get; set; } = new ObjectHeader();
|
||||
/// Object that receives this damage.
|
||||
public ObjectHeader Dmg_target { get; set; } = new ObjectHeader();
|
||||
/// Object that deals this damage.
|
||||
public ObjectHeader Dmg_inflicter { get; set; } = new ObjectHeader();
|
||||
/// Inflicted damage ID.
|
||||
public uint Damage_id { get; set; } = 0;
|
||||
/// How much damage was inflicted.
|
||||
public int Dmg_amount { get; set; } = 0;
|
||||
/// New HP.
|
||||
public uint New_hp { get; set; } = 0;
|
||||
/// Hitbox ID (?).
|
||||
public uint Hitbox_id { get; set; } = 0;
|
||||
/// Hit x position.
|
||||
public Half X_pos { get; set; } = 0;
|
||||
/// Hit y position.
|
||||
public Half Y_pos { get; set; } = 0;
|
||||
/// Hit z position.
|
||||
public Half Z_pos { get; set; } = 0;
|
||||
public ushort unk1 { get; set; } = 0;
|
||||
public ushort unk2 { get; set; } = 0;
|
||||
public ushort unk3 { get; set; } = 0;
|
||||
public ushort unk4 { get; set; } = 0;
|
||||
public ushort unk5 { get; set; } = 0;
|
||||
public ushort unk6 { get; set; } = 0;
|
||||
public ushort unk7 { get; set; } = 0;
|
||||
|
||||
public EnemyKilledPacket()
|
||||
{
|
||||
@ -18,12 +45,13 @@ namespace PSO2SERVER.Protocol.Packets
|
||||
public override byte[] Build()
|
||||
{
|
||||
var pkt = new PacketWriter();
|
||||
pkt.Write(X_pos);
|
||||
return pkt.ToArray();
|
||||
}
|
||||
|
||||
public override PacketHeader GetHeader()
|
||||
{
|
||||
return new PacketHeader(0x04, 0x0F, PacketFlags.None);
|
||||
return new PacketHeader(0x04, 0x0F, PacketFlags.OBJECT_RELATED);
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
@ -27,42 +27,42 @@ namespace PSO2SERVER.Protocol.Packets
|
||||
/// CharacterSpawnPacket Struct
|
||||
/// </summary>
|
||||
/// Spawned character's player object.
|
||||
public ObjectHeader objHeader { get; set; } = new ObjectHeader();
|
||||
public PSOLocation objPosition { get; set; } = new PSOLocation();
|
||||
public ushort unk1 { get; set; } = 0;
|
||||
public ObjectHeader ObjHeader { get; set; } = new ObjectHeader();
|
||||
public PSOLocation ObjPosition { get; set; } = new PSOLocation();
|
||||
public ushort Unk1 { get; set; } = 0;
|
||||
/// Always `Character`. (?)
|
||||
public string objName { get; set; } = "Character";//0x20
|
||||
public ushort unk3 { get; set; } = 1;
|
||||
public ushort unk4 { get; set; }
|
||||
public uint unk5 { get; set; }
|
||||
public uint unk6 { get; set; }
|
||||
public uint unk7 { get; set; }
|
||||
public uint unk8 { get; set; }
|
||||
public CharacterSpawnType spawn_type { get; set; }
|
||||
public byte unk9 { get; set; }
|
||||
public ushort unk10 { get; set; }
|
||||
public Character _character { get; set; }
|
||||
public uint unk11 { get; set; }
|
||||
public string ObjName { get; set; } = "Character";//0x20
|
||||
public ushort Unk3 { get; set; } = 1;
|
||||
public ushort Unk4 { get; set; }
|
||||
public uint Unk5 { get; set; }
|
||||
public uint Unk6 { get; set; }
|
||||
public uint Unk7 { get; set; }
|
||||
public uint Unk8 { get; set; }
|
||||
public CharacterSpawnType SpawnType { get; set; }
|
||||
public byte Unk9 { get; set; }
|
||||
public ushort Unk10 { get; set; }
|
||||
public Character Character { get; set; }
|
||||
public uint Unk11 { get; set; }
|
||||
/// Set to `1` if the player is a GM.
|
||||
public uint gm_flag { get; set; }
|
||||
public string nickname { get; set; }
|
||||
public uint GmFlag { get; set; }
|
||||
public string Nickname { get; set; }
|
||||
//#[SeekAfter(0x60)]
|
||||
public byte[] unk12 { get; set; } = new byte[0x40];
|
||||
public byte[] Unk12 { get; set; } = new byte[0x40];
|
||||
|
||||
public CharacterSpawnPacket(Character character, PSOLocation locatiion)
|
||||
{
|
||||
objHeader = new ObjectHeader((uint)character.Account.AccountId, ObjectType.Player);
|
||||
_character = character;
|
||||
objPosition = locatiion;
|
||||
ObjHeader = new ObjectHeader((uint)character.Account.AccountId, ObjectType.Player);
|
||||
Character = character;
|
||||
ObjPosition = locatiion;
|
||||
}
|
||||
|
||||
public CharacterSpawnPacket(Character character, PSOLocation locatiion, bool isme, bool isgm)
|
||||
{
|
||||
objHeader = new ObjectHeader((uint)character.Account.AccountId, ObjectType.Player);
|
||||
_character = character;
|
||||
spawn_type = isme ? CharacterSpawnType.Myself : CharacterSpawnType.Other;
|
||||
objPosition = locatiion;
|
||||
gm_flag = isgm ? (uint)1 : 0;
|
||||
ObjHeader = new ObjectHeader((uint)character.Account.AccountId, ObjectType.Player);
|
||||
Character = character;
|
||||
SpawnType = isme ? CharacterSpawnType.Myself : CharacterSpawnType.Other;
|
||||
ObjPosition = locatiion;
|
||||
GmFlag = isgm ? (uint)1 : 0;
|
||||
}
|
||||
|
||||
#region implemented abstract members of Packet
|
||||
@ -71,25 +71,25 @@ namespace PSO2SERVER.Protocol.Packets
|
||||
{
|
||||
var pkt = new PacketWriter();
|
||||
// Accounts header
|
||||
pkt.WriteStruct(objHeader);
|
||||
pkt.WriteStruct(ObjHeader);
|
||||
// Spawn position
|
||||
pkt.WritePosition(objPosition);
|
||||
pkt.Write(unk1);
|
||||
pkt.WriteFixedLengthASCII(objName, 0x20);
|
||||
pkt.Write(unk3); // 0x44
|
||||
pkt.Write(unk4);
|
||||
pkt.Write(unk5);
|
||||
pkt.Write(unk6);
|
||||
pkt.Write(unk7);
|
||||
pkt.Write(unk8);
|
||||
pkt.Write((byte)spawn_type);
|
||||
pkt.Write(unk9);
|
||||
pkt.Write(unk10);
|
||||
pkt.WritePosition(ObjPosition);
|
||||
pkt.Write(Unk1);
|
||||
pkt.WriteFixedLengthASCII(ObjName, 0x20);
|
||||
pkt.Write(Unk3); // 0x44
|
||||
pkt.Write(Unk4);
|
||||
pkt.Write(Unk5);
|
||||
pkt.Write(Unk6);
|
||||
pkt.Write(Unk7);
|
||||
pkt.Write(Unk8);
|
||||
pkt.Write((byte)SpawnType);
|
||||
pkt.Write(Unk9);
|
||||
pkt.Write(Unk10);
|
||||
// Character data.
|
||||
pkt.Write(_character.BuildCharacterByteArray());
|
||||
pkt.Write(unk11);
|
||||
pkt.Write(gm_flag);
|
||||
pkt.WriteFixedLengthUtf16(_character.Account.Nickname, 0x10);
|
||||
pkt.Write(Character.BuildCharacterByteArray());
|
||||
pkt.Write(Unk11);
|
||||
pkt.Write(GmFlag);
|
||||
pkt.WriteFixedLengthUtf16(Character.Account.Nickname, 0x10);
|
||||
//for (var i = 0; i < 0x60; i++)
|
||||
// pkt.Write((byte)0);
|
||||
//pkt.Write(unk12);
|
||||
|
@ -135,6 +135,9 @@
|
||||
<Reference Include="System.Diagnostics.DiagnosticSource, Version=8.0.0.1, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51, processorArchitecture=MSIL">
|
||||
<HintPath>..\packages\System.Diagnostics.DiagnosticSource.8.0.1\lib\net462\System.Diagnostics.DiagnosticSource.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="System.Half, Version=1.0.0.0, Culture=neutral, processorArchitecture=MSIL">
|
||||
<HintPath>..\packages\Half.1.0.0\lib\netstandard2.0\System.Half.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="System.IO.Pipelines, Version=8.0.0.0, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51, processorArchitecture=MSIL">
|
||||
<HintPath>..\packages\System.IO.Pipelines.8.0.0\lib\net462\System.IO.Pipelines.dll</HintPath>
|
||||
</Reference>
|
||||
@ -165,6 +168,7 @@
|
||||
<Compile Include="ConsoleSystem.cs" />
|
||||
<Compile Include="Crypto\KeyLoader.cs" />
|
||||
<Compile Include="Logger.cs" />
|
||||
<Compile Include="Models\BattleStats.cs" />
|
||||
<Compile Include="Models\BlockInfo.cs" />
|
||||
<Compile Include="Models\CharacterAddtionStruct.cs" />
|
||||
<Compile Include="Models\FixedTypes.cs" />
|
||||
@ -178,6 +182,7 @@
|
||||
<Compile Include="Models\PSOPalette.cs" />
|
||||
<Compile Include="Models\Quest.cs" />
|
||||
<Compile Include="Models\RevealedRegions.cs" />
|
||||
<Compile Include="Models\Stats.cs" />
|
||||
<Compile Include="Models\Time.cs" />
|
||||
<Compile Include="Network\Ipv4Addr.cs" />
|
||||
<Compile Include="Network\PortChecker.cs" />
|
||||
|
@ -4,6 +4,7 @@
|
||||
<package id="BouncyCastle.Cryptography" version="2.4.0" targetFramework="net48" />
|
||||
<package id="EntityFramework" version="6.5.1" targetFramework="net48" />
|
||||
<package id="Google.Protobuf" version="3.28.1" targetFramework="net48" />
|
||||
<package id="Half" version="1.0.0" targetFramework="net48" />
|
||||
<package id="K4os.Compression.LZ4" version="1.3.8" targetFramework="net48" />
|
||||
<package id="K4os.Compression.LZ4.Streams" version="1.3.8" targetFramework="net48" />
|
||||
<package id="K4os.Hash.xxHash" version="1.0.8" targetFramework="net48" />
|
||||
|
Loading…
Reference in New Issue
Block a user