Jump to content

Recommended Posts

Posted

In this guide we will cover how to understand and implement the packages belonging to the Lineage 2 UI. More specifically the Essence version UI.

Required UTPT (Unreal Tournament Package Tool) : https://github.com/l2jsourcecode/ut-package-tool
File InterfaceClassic.U
 

The packages are embedded into the interface, specifically located in the InterfaceClassic.u file. Some key classes include:

  • UIPacket.uc (defines package structure)
  • UIProtocol.uc (defines package opcode)

 


The packages are split into two main types:

  • Client to Server (prefix: C)
  • Server to Client (prefix: S)



For example, in the UIProtocol class:

 

const C_EX_COSTUME_LOCK = 573; // packet from client to server
const S_EX_COSTUME_LOCK = 785; // packet from server to client

 

We’ll use this package as a case study to delve deeper.

Understanding Client to Server Packets Let's examine a packet from the client with an opcode of 573. On the server side, opcodes are converted into hexadecimal values using a formula:

opcode - C_MAX
 

where C_MAX is an opcode defined in UIProtocol. Recently, C_MAX has been set to 209.
For instance
:

573 - 209 = 364 364 -> hex = 16C

This hexadecimal value (16C) corresponds to the opcode for the Java class.
Understanding Server to Client Packets
For server packets, we also use hexadecimal conversion but with a different opcode:

S_MAX = 255
S_MAX is applied to packages with an opcode above 255. For values below this, simply convert the number to hex.



For example:

const S_EX_COSTUME_LOCK = 785; 785 - 255 = 530 530 -> hex = 212 = 0xFE:0x212
Here, 0xFE represents S_MAX.



Packet Structure


Finally
, let’s look at the packet structure.

Open UIPacket
.uc and locate the C_EX_COSTUME_LOCK to examine its structure in detail.
 
struct _C_EX_COSTUME_LOCK
{
    var int nCostumeId;
    var int nLockState;
};
 
We observe two integer variables, representing the structure of the package transmitted to the server. By continuing our search within this file, we find the following code:
 
static function bool Encode_C_EX_COSTUME_LOCK(out array<byte> stream, _C_EX_COSTUME_LOCK packet)
{
    if(!EncodeInt(stream, packet.nCostumeId)) {
        return false;
    }
    if(!EncodeChar(stream, packet.nLockState)) {
        return false;
    }
    return true;
}

 

This function encodes the packet structure for transmission. As discussed earlier, the data types within the packet are crucial. Specifically:

 

EncodeInt corresponds to an integer. EncodeChar is treated as a short in Java, typically read using readC, while readD is used for EncodeInt. Having examined the client-to-server packet structure, let's now explore the structure expected by the client from the server. The process is similar. We locate _S_EX_COSTUME_LOCK within UIPacket:

 

 

struct _S_EX_COSTUME_LOCK
{
    var byte bResult;
    var int nCostumeId;
    var int nLockState;
};

 


Data Decryption

The corresponding decryption function is:

 

static function bool Decode_S_EX_COSTUME_LOCK(out _S_EX_COSTUME_LOCK packet)
{
    if(!DecodeBool(packet.bResult)) {
        return false;
    }
    if(!DecodeInt(packet.nCostumeId)) {
        return false;
    }
    if(!DecodeChar(packet.nLockState)) {
        return false;
    }
    return true;
}

 

Here, the variables are decoded as follows:

 

bResult is decoded as a Bool.

nCostumeId is decoded as an Int.

nLockState is decoded as a Char.

 

This consistent approach to encoding and decoding ensures reliable data transmission between the client and server.



Example usage
https://imgur.com/a/zVZHVzq


Credits : L2j West Coast Custom ( https://discord.gg/n2c8meMN )

Create an account or sign in to comment

You need to be a member in order to leave a comment

Create an account

Sign up for a new account in our community. It's easy!

Register a new account

Sign in

Already have an account? Sign in here.

Sign In Now


  • Posts

    • I genuinely admire your bravery - in an age where AI can whip up something better in under a minute, you still stubbornly try to sell these "projects" of yours on a forum that’s been clinically dead for years. That’s no longer determination, that’s digital archaeology. I just can’t tell whether you’re actually trying to make money, or simply testing how much we can endure before we ask an AI to generate you some actual talent.   And ofc AI will make it for free, $220 saved.
    • I’m glad I’m not the only one who appreciates Maxthor’s involvement in group gay orgies, he can’t be bothered to reply to messages, but covering the entire forum in gay lights is absolutely no issue for him. As for the project - the forum is packed with feedback from the testers, the lads are spending every spare moment fixing even the tiniest typo in an NPC’s text. I’ll share the links as soon as I get the green light. Edit: I forgot to add that the GM recruitment will begin once the links are released. Three people will be accepted, and they’ll work in a three-shift rotation so that there’s always a GM available online.
    • Added: a brand-new default dashboard template. You can now add multiple game/login server builds. Full support for running both PTS & L2J servers simultaneously, with switching between them. Payment systems: added OmegaPay and Pally (new PayPal-style API). Account history now stores everything: donations, items delivered to characters, referrals, transfers between game accounts, and coin transfers to another master account. Personal Promo Code System: you can create a promo code and assign it to a user or promoter. When donating, a player can enter this promo code to receive bonus coins, and the promo code owner also receives a bonus — all fully configurable in the admin panel.     Look demo site: demo
    • One of best project i play last few years
    • for me no  😕   https://files.fm/f/jewqgu9fkb
  • Topics

×
×
  • Create New...

AdBlock Extension Detected!

Our website is made possible by displaying online advertisements to our members.

Please disable AdBlock browser extension first, to be able to use our community.

I've Disabled AdBlock