using System; using System.IO; using System.Linq; using System.Runtime.InteropServices; using System.Text; using static PSO2SERVER.Models.Character; namespace PSO2SERVER.Models { [StructLayout(LayoutKind.Sequential)] public unsafe struct ShortItemId { byte ItemType; byte Id; ushort Subid; } [StructLayout(LayoutKind.Sequential)] public unsafe struct ItemId { ushort ItemType; ushort Id; ushort Unk3; ushort Subid; } [StructLayout(LayoutKind.Sequential)] public struct PSO2Items { long guid; ItemId id; Items data; } [StructLayout(LayoutKind.Explicit)] public struct Items { [FieldOffset(0)] public PSO2ItemWeapon Weapon; [FieldOffset(0)] public PSO2ItemClothing Clothing; [FieldOffset(0)] public PSO2ItemConsumable Consumable; [FieldOffset(0)] public PSO2ItemCamo Camo; [FieldOffset(0)] public PSO2ItemUnit Unit; //[FieldOffset(0)] //public byte[] Unknown; //[FieldOffset(0)] //public PSO2ItemNone None; } [StructLayout(LayoutKind.Sequential)] public unsafe struct PSO2ItemNone { } [StructLayout(LayoutKind.Sequential)] public unsafe struct PSO2ItemWeapon { byte flags; byte element; byte force; byte grind; byte grindPercent; byte unknown1; short unknown2; fixed short affixes[8]; uint potential; byte extend; byte unknown3; ushort unknown4; uint unknown5; uint unknown6; } [StructLayout(LayoutKind.Sequential)] public unsafe struct PSO2ItemClothing { ushort flags; fixed byte unk1[0x14]; public HSVColor Color; fixed byte unk2[0xA]; ushort Unk3; } [StructLayout(LayoutKind.Sequential)] public unsafe struct PSO2ItemConsumable { ushort flags; fixed byte unk1[0x24]; ushort amount; } [StructLayout(LayoutKind.Sequential)] public unsafe struct PSO2ItemCamo { byte unk1; byte unk2; byte unk3; fixed byte unk4[0x24]; byte unk5; } [StructLayout(LayoutKind.Sequential)] public unsafe struct PSO2ItemUnit { byte flags; byte EnhLevel; byte EnhPercent; byte Unk1; // 使用 fixed 数组来存储附加信息 fixed ushort Affixes[8]; // Item affix IDs (0 to 4095) fixed byte unk4[0x7]; uint Potential; // 使用 fixed 数组来存储未知字段 fixed byte Unk2[4]; uint Unk3; ushort Unk4; ushort Unk5; // 提供访问固定数组的属性 Span AffixSpan { get { fixed (ushort* p = Affixes) { return new Span(p, 8); } } } } public struct Campaign { /// Campaign ID. public uint Id; // 对应 Rust 的 u32 /// Start timestamp. public TimeSpan StartDate; // 对应 Rust 的 Duration /// End timestamp. public TimeSpan EndDate; // 对应 Rust 的 Duration /// Campaign title (固定长度 0x3E). private const int TitleLength = 0x3E; private byte[] titleBytes; public string Title { get => Encoding.ASCII.GetString(titleBytes).TrimEnd('\0'); set { titleBytes = new byte[TitleLength]; byte[] valueBytes = Encoding.ASCII.GetBytes(value); Array.Copy(valueBytes, titleBytes, Math.Min(valueBytes.Length, TitleLength)); } } /// Campaign conditions (固定长度 0x102). private const int ConditionsLength = 0x102; private byte[] conditionsBytes; public string Conditions { get => Encoding.ASCII.GetString(conditionsBytes).TrimEnd('\0'); set { conditionsBytes = new byte[ConditionsLength]; byte[] valueBytes = Encoding.ASCII.GetBytes(value); Array.Copy(valueBytes, conditionsBytes, Math.Min(valueBytes.Length, ConditionsLength)); } } // 从数据流中读取 Campaign public static Campaign FromStream(Stream stream) { using (BinaryReader reader = new BinaryReader(stream, Encoding.ASCII, true)) { Campaign campaign = new Campaign { Id = reader.ReadUInt32(), StartDate = TimeSpan.FromTicks(reader.ReadInt64()), EndDate = TimeSpan.FromTicks(reader.ReadInt64()), }; campaign.titleBytes = reader.ReadBytes(TitleLength); campaign.conditionsBytes = reader.ReadBytes(ConditionsLength); return campaign; } } // 将 Campaign 写入数据流 public void WriteToStream(Stream stream) { using (BinaryWriter writer = new BinaryWriter(stream, Encoding.ASCII, true)) { writer.Write(Id); writer.Write(StartDate.Ticks); writer.Write(EndDate.Ticks); writer.Write(titleBytes); writer.Write(conditionsBytes); } } } public enum ItemTypes { NoItem, Weapon, Clothing, Consumable, Camo, Unit, Unknown } [Flags] public enum ItemFlags { Locked = 0x01, BoundToOwner = 0x02 } public enum ItemElement { None, Fire, Ice, Lightning, Wind, Light, Dark } public class PSO2Item { public const int Size = 0x38; MemoryStream stream; //TODO ItemTypes type = ItemTypes.Consumable; byte[] data = new byte[Size]; public override string ToString() { return string.Format("Data: {0:X}", BitConverter.ToString(data)).Replace('-', ' '); } public PSO2Item(byte[] data) { SetData(data); } public byte[] GetData() { return data; } public void SetData(byte[] data) { this.data = data; stream = new MemoryStream(data, true); } public long GetGUID() { byte[] guid = new byte[sizeof(long)]; stream.Seek(0, SeekOrigin.Begin); stream.Read(guid, 0, sizeof(long)); return BitConverter.ToInt64(guid, 0); } public void SetGUID(long guid) { stream.Seek(0, SeekOrigin.Begin); stream.Write(BitConverter.GetBytes(guid), 0, 8); } public int[] GetID() { byte[] ID = new byte[sizeof(int)]; byte[] subID = new byte[sizeof(int)]; stream.Seek(0, SeekOrigin.Begin); stream.Read(ID, 0x08, sizeof(int)); stream.Read(subID, 0x0C, sizeof(int)); return new int[] { BitConverter.ToInt32(ID, 0), BitConverter.ToInt32(subID, 0) }; } public void SetID(int ID, int subID) { stream.Seek(0, SeekOrigin.Begin); stream.Write(BitConverter.GetBytes(ID), 0x08, sizeof(int)); stream.Write(BitConverter.GetBytes(subID), 0x0C, sizeof(int)); } // ... } } public static class AffixUtils { public static ushort[] ReadPackedAffixes(Stream reader) { byte[] packed = new byte[12]; reader.Read(packed, 0, packed.Length); ushort[] affixes = new ushort[8]; for (int i = 0; i < 4; i++) { affixes[i * 2] = BitConverter.ToUInt16(new byte[] { packed[i * 3], (byte)((packed[i * 3 + 2] & 0xF0) >> 4) }, 0); affixes[i * 2 + 1] = BitConverter.ToUInt16(new byte[] { packed[i * 3 + 1], (byte)(packed[i * 3 + 2] & 0xF) }, 0); } return affixes; } public static void WritePackedAffixes(ushort[] affixes, Stream writer) { byte[] packed = new byte[12]; for (int i = 0; i < 4; i++) { byte[] affix1 = BitConverter.GetBytes(affixes[i * 2]); byte[] affix2 = BitConverter.GetBytes(affixes[i * 2 + 1]); packed[i * 3] = affix1[0]; packed[i * 3 + 1] = affix2[0]; packed[i * 3 + 2] = (byte)((affix1[1] << 4) | (affix2[1] & 0xF)); } writer.Write(packed, 0, packed.Length); } } public class PacketError : Exception { public string PacketName { get; } public string FieldName { get; } public PacketError(string packetName, string fieldName, Exception innerException) : base($"Error in packet '{packetName}', field '{fieldName}'", innerException) { PacketName = packetName; FieldName = fieldName; } }