PSO2SERVER/Server/Packets/Handlers/MovementHandlers.cs
2024-09-11 17:13:00 +08:00

329 lines
12 KiB
C#

using PSO2SERVER.Models;
using PSO2SERVER.Packets.PSOPackets;
using System;
using System.Runtime.InteropServices;
namespace PSO2SERVER.Packets.Handlers
{
[PacketHandlerAttr(0x04, 0x07)]
public class MovementHandler : PacketHandler
{
#region implemented abstract members of PacketHandler
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
PacketReader reader = new PacketReader(data);
// This packet is "Compressed" basically.
reader.ReadBytes(6); // Get past the junk
// For simplicity's sake, read the 3 flag bytes into a big int
byte[] flagBytes = reader.ReadBytes(3);
uint dataFlags = flagBytes[0];
dataFlags |= (uint)(flagBytes[1] << 8);
dataFlags |= (uint)(flagBytes[2] << 16);
PackedData theFlags = (PackedData)dataFlags;
// Debug
Logger.WriteInternal("[MOV] Movement packet from {0} contains {1} data.", context.Character.Name, theFlags);
// TODO: Maybe do this better someday
FullMovementData dstData = new FullMovementData();
if (theFlags.HasFlag(PackedData.ENT1_ID))
{
dstData.entity1.ID = (uint)reader.ReadUInt64();
}
if (theFlags.HasFlag(PackedData.ENT1_TYPE))
{
dstData.entity1.EntityType = (EntityType)reader.ReadUInt16();
}
if (theFlags.HasFlag(PackedData.ENT1_A))
{
dstData.entity1.Unknown_A = reader.ReadUInt16();
}
if (theFlags.HasFlag(PackedData.ENT2_ID))
{
dstData.entity1.ID = (uint)reader.ReadUInt64();
}
if (theFlags.HasFlag(PackedData.ENT2_TYPE))
{
dstData.entity1.EntityType = (EntityType)reader.ReadUInt16();
}
if (theFlags.HasFlag(PackedData.ENT2_A))
{
dstData.entity1.Unknown_A = reader.ReadUInt16();
}
if (theFlags.HasFlag(PackedData.TIMESTAMP))
{
dstData.timestamp = reader.ReadUInt32();
context.MovementTimestamp = dstData.timestamp;
}
if (theFlags.HasFlag(PackedData.ROT_X))
{
dstData.rotation.x = reader.ReadUInt16();
context.CurrentLocation.RotX = Helper.FloatFromHalfPrecision(dstData.rotation.x);
}
if (theFlags.HasFlag(PackedData.ROT_Y))
{
dstData.rotation.y = reader.ReadUInt16();
context.CurrentLocation.RotY = Helper.FloatFromHalfPrecision(dstData.rotation.y);
}
if (theFlags.HasFlag(PackedData.ROT_Z))
{
dstData.rotation.z = reader.ReadUInt16();
context.CurrentLocation.RotZ = Helper.FloatFromHalfPrecision(dstData.rotation.z);
}
if (theFlags.HasFlag(PackedData.ROT_W))
{
dstData.rotation.w = reader.ReadUInt16();
context.CurrentLocation.RotW = Helper.FloatFromHalfPrecision(dstData.rotation.w);
}
if (theFlags.HasFlag(PackedData.CUR_X))
{
dstData.currentPos.x = reader.ReadUInt16();
context.CurrentLocation.PosX = Helper.FloatFromHalfPrecision(dstData.currentPos.x);
}
if (theFlags.HasFlag(PackedData.CUR_Y))
{
dstData.currentPos.y = reader.ReadUInt16();
context.CurrentLocation.PosY = Helper.FloatFromHalfPrecision(dstData.currentPos.y);
}
if (theFlags.HasFlag(PackedData.CUR_Z))
{
dstData.currentPos.z = reader.ReadUInt16();
context.CurrentLocation.PosZ = Helper.FloatFromHalfPrecision(dstData.currentPos.z);
}
if (theFlags.HasFlag(PackedData.UNKNOWN4))
{
dstData.Unknown2 = reader.ReadUInt16();
}
if (theFlags.HasFlag(PackedData.UNK_X))
{
dstData.unknownPos.x = reader.ReadUInt16();
context.LastLocation.PosX = Helper.FloatFromHalfPrecision(dstData.unknownPos.x);
}
if (theFlags.HasFlag(PackedData.UNK_Y))
{
dstData.unknownPos.y = reader.ReadUInt16();
context.LastLocation.PosY = Helper.FloatFromHalfPrecision(dstData.unknownPos.y);
}
if (theFlags.HasFlag(PackedData.UNK_Z))
{
dstData.unknownPos.z = reader.ReadUInt16();
context.LastLocation.PosZ = Helper.FloatFromHalfPrecision(dstData.unknownPos.z);
}
if (theFlags.HasFlag(PackedData.UNKNOWN5))
{
dstData.Unknown3 = reader.ReadUInt16();
}
if (theFlags.HasFlag(PackedData.UNKNOWN6))
{
if (theFlags.HasFlag(PackedData.UNKNOWN7))
{
dstData.Unknown4 = reader.ReadByte();
}
else
{
dstData.Unknown4 = reader.ReadUInt32();
}
}
Logger.WriteInternal("[MOV] Player moving! {0} -> ({1}, {2}, {3})", context.Character.Name, context.CurrentLocation.PosX,
context.CurrentLocation.PosY, context.CurrentLocation.PosZ);
foreach (var c in Server.Instance.Clients)
{
if (c.Character == null || c == context || c.CurrentZone != context.CurrentZone)
continue;
c.SendPacket(0x4, 0x7, flags, data);
}
}
#endregion
}
[PacketHandlerAttr(0x04, 0x71)]
public class MovementEndHandler : PacketHandler
{
#region implemented abstract members of PacketHandler
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
PacketReader reader = new PacketReader(data);
FullMovementData movData = reader.ReadStruct<FullMovementData>();
if (movData.entity1.ID == 0 && movData.entity2.ID != 0)
movData.entity1 = movData.entity2;
movData.timestamp = 0;
// This could be simplified
PacketWriter writer = new PacketWriter();
writer.WriteStruct(movData);
Logger.WriteInternal("[MOV] {0} stopped moving at ({1}, {2}, {3})", context.Character.Name,
Helper.FloatFromHalfPrecision(movData.currentPos.x), Helper.FloatFromHalfPrecision(movData.currentPos.y),
Helper.FloatFromHalfPrecision(movData.currentPos.z));
foreach (var c in Server.Instance.Clients)
{
if (c == context || c.Character == null || c.CurrentZone != context.CurrentZone)
continue;
c.SendPacket(0x04, 0x71, 0x40, writer.ToArray());
}
}
#endregion
}
[PacketHandlerAttr(0x4, 0x8)]
public class MovementActionHandler : PacketHandler
{
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
PacketReader reader = new PacketReader(data);
reader.ReadStruct<ObjectHeader>(); // Skip blank entity header.
var preformer = reader.ReadStruct<ObjectHeader>(); // Preformer
byte[] preData = reader.ReadBytes(40);
string command = reader.ReadAscii(0x922D, 0x45);
byte[] rest = reader.ReadBytes(4);
uint thingCount = reader.ReadMagic(0x922D, 0x45);
byte[] things;
PacketWriter thingWriter = new PacketWriter();
for (int i = 0; i < thingCount; i++)
{
thingWriter.Write(reader.ReadBytes(4));
}
things = thingWriter.ToArray();
byte[] final = reader.ReadBytes(4);
Logger.WriteInternal("[ACT] {0} is preforming {1}", context.Character.Name, command);
foreach (var c in Server.Instance.Clients)
{
if (c == context || c.Character == null || c.CurrentZone != context.CurrentZone)
continue;
PacketWriter output = new PacketWriter();
output.WriteStruct(new ObjectHeader((uint)context.User.PlayerId, EntityType.Player));
output.WriteStruct(preformer);
output.Write(preData);
output.WriteAscii(command, 0x4315, 0x7A);
output.Write(rest);
output.WriteMagic(thingCount, 0x4315, 0x7A);
output.Write(things);
output.Write(final);
c.SendPacket(0x4, 0x80, 0x44, output.ToArray());
}
}
}
[PacketHandlerAttr(0x4, 0x3C)]
public class ActionUpdateHandler : PacketHandler
{
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
PacketReader reader = new PacketReader(data);
reader.ReadStruct<ObjectHeader>(); // Read the blank
ObjectHeader actor = reader.ReadStruct<ObjectHeader>(); // Read the actor
byte[] rest = reader.ReadBytes(32); // TODO Map this out and do stuff with it!
foreach (var c in Server.Instance.Clients)
{
if (c == context || c.Character == null || c.CurrentZone != context.CurrentZone)
continue;
PacketWriter writer = new PacketWriter();
writer.WriteStruct(new ObjectHeader((uint)c.User.PlayerId, EntityType.Player));
writer.WriteStruct(actor);
writer.Write(rest);
c.SendPacket(0x4, 0x81, 0x40, writer.ToArray());
}
}
}
[Flags]
public enum PackedData : Int32
{
ENT1_ID = 1,
ENT1_TYPE = 2,
ENT1_A = 4,
ENT2_ID = 8,
ENT2_TYPE = 0x10,
ENT2_A = 0x20,
TIMESTAMP = 0x40,
ROT_X = 0x80,
ROT_Y = 0x100,
ROT_Z = 0x200,
ROT_W = 0x400,
CUR_X = 0x800,
CUR_Y = 0x1000,
CUR_Z = 0x2000,
UNKNOWN4 = 0x4000,
UNK_X = 0x8000,
UNK_Y = 0x10000,
UNK_Z = 0x20000,
UNKNOWN5 = 0x40000,
UNKNOWN6 = 0x80000,
UNKNOWN7 = 0x100000
}
public struct PackedVec4
{
public UInt16 x, y, z, w;
public PackedVec4(PSOLocation location)
{
this.x = Helper.FloatToHalfPrecision(location.RotX);
this.y = Helper.FloatToHalfPrecision(location.RotY);
this.z = Helper.FloatToHalfPrecision(location.RotZ);
this.w = Helper.FloatToHalfPrecision(location.RotW);
}
}
public struct PackedVec3
{
public UInt16 x, y, z;
public PackedVec3(PSOLocation location)
{
this.x = Helper.FloatToHalfPrecision(location.PosX);
this.y = Helper.FloatToHalfPrecision(location.PosY);
this.z = Helper.FloatToHalfPrecision(location.PosZ);
}
}
[StructLayout(LayoutKind.Explicit, Size = 0x38)]
public struct FullMovementData
{
[FieldOffset(0x0)]
public ObjectHeader entity1;
[FieldOffset(0xC)]
public ObjectHeader entity2;
[FieldOffset(0x18)]
public UInt32 timestamp;
[FieldOffset(0x1C)]
public PackedVec4 rotation;
[FieldOffset(0x24)]
public PackedVec3 currentPos;
[FieldOffset(0x2A)]
public UInt16 Unknown2; // This MAY be part of lastPos, as lastPos may be a Vec4?
[FieldOffset(0x2C)]
public PackedVec3 unknownPos;
[FieldOffset(0x32)]
public UInt16 Unknown3; // This MAY be part of currentPos, as lastPos may be a Vec4?
[FieldOffset(0x34)]
public UInt32 Unknown4;
}
}