Skip to content

Synchronization between Server and Client

In RPGJS, synchronization between the server and the client is a crucial aspect to ensure a consistent and immersive gameplay experience. This readme will guide you through the process of synchronizing data between the server and the client, utilizing schemas and hooks provided by RPGJS.

Understanding RPGJS Schemas and Data Synchronization

RPGJS uses schemas to define the structure of data that needs to be synchronized between the server and the client. These schemas ensure that the properties of entities, such as players and items, are consistent and correctly communicated between the server and the client. When a property changes in a schema, it is automatically sent to the client, keeping the game state up-to-date.

Schemas work within the context of a "room" in RPGJS, where a room is essentially a map in the game world. It's important to note that when a player changes maps, synchronization can be lost if not handled properly.

Best Practices for Synchronization

1. Avoid Synchronization Issues After Changing Maps

When a player changes maps, it's crucial not to continue executing the code in the onConnected method immediately. This is because the properties might not be synchronized with the client yet. Instead, use the onJoinMap hook to ensure synchronization before performing any actions.

typescript
import { RpgPlayer, RpgPlayerHooks } from '@rpgjs/server';

export const player: RpgPlayerHooks = {
    async onConnected(player: RpgPlayer) {
        player.hitbox(32, 16);
        player.setGraphic('hero');
        await player.changeMap('medieval', {
            x: 100,
            y: 100
        });
    },
    // Synchronization is ensured in this hook
    onJoinMap(player: RpgPlayer) {
        player.hp = 500;
    }
};

2. Alternative Synchronization Method

Another approach is to access the player instance from the RpgWorld and set properties directly for synchronization purposes. This can be helpful when you need to ensure synchronization before performing certain actions.

typescript
import { RpgPlayer, RpgPlayerHooks, RpgWorld } from '@rpgjs/server';

export const player: RpgPlayerHooks = {
    async onConnected(player: RpgPlayer) {
        player.hitbox(32, 16);
        player.setGraphic('hero');
        await player.changeMap('medieval', {
            x: 100,
            y: 100
        });
        // Access the player instance from RpgWorld for synchronization
        const currentPlayer = RpgWorld.getPlayer(player.id);
        currentPlayer.hp = 10;
    }
};

3. Custom Synchronization with Schemas

You can also create custom synchronization by extending existing schemas or defining new ones.

Example:

typescript
import { RpgPlayerHooks, RpgPlayer } from '@rpgjs/server';

declare module '@rpgjs/server' {
    export interface RpgPlayer {
        bronze: number;
    }
}

export const player: RpgPlayerHooks = {
    props: {
        bronze: Number // Add the custom property
    },
    onConnected(player: RpgPlayer) {
        player.bronze = 0; // Initialize the custom property
    }
};

With these steps, you've successfully added a custom property named bronze to the player entity and ensured its synchronization with the client. Any changes you make to this property will be automatically sent to the client, maintaining a consistent game state for all players.

Save Custom Props

In addition to synchronization, RPGJS provides a mechanism to save snapshots of player data, including custom properties, to databases.

ts
import { RpgPlayerHooks, RpgPlayer } from '@rpgjs/server';

declare module '@rpgjs/server' {
    export interface RpgPlayer {
        bronze: number;
    }
}

export const player: RpgPlayerHooks = {
    props: {
        bronze: Number
    },
    onConnected(player: RpgPlayer) {
        player.bronze = 0;
    },
    onDisconnect(player: RpgPlayer) {
        // Save the player data, including custom properties, to the database
        const json = player.save();
        console.log(json) // --> { ..., "bronze": 0 }
    }
}

Save Custom Props but not synchronized with the client

ts
import { RpgPlayerHooks, RpgPlayer } from '@rpgjs/server';

declare module '@rpgjs/server' {
    export interface RpgPlayer {
        secret: string;
    }
}

export const player: RpgPlayerHooks = {
    props: {
        secret: {
            $syncWithClient: false
        }
    },
    onConnected(player: RpgPlayer) {
        player.secret = 'mysecretvalue';
        const json = player.save();
        console.log(json) // --> { ..., "secret": "mysecretvalue" }
    }
}

The custom property secret could be used for various purposes in your game. It might hold sensitive information that shouldn't be shared with the client, such as a player's authentication token, API keys, or any other private data.

By configuring the property with $syncWithClient: false, you can control what data is sent to the client and what remains hidden on the server. This enables you to strike a balance between synchronization and data security.

Synchronize with client but do not register

ts
import { RpgPlayerHooks, RpgPlayer } from '@rpgjs/server';

declare module '@rpgjs/server' {
    export interface RpgPlayer {
        message: string;
    }
}

export const player: RpgPlayerHooks = {
    props: {
        message: {
            $permanent: false
        }
    },
    onConnected(player: RpgPlayer) {
        player.message = 'custom message';
        const json = player.save();
        console.log(json); // --> { ... } // does not include the message property
    }
}

The $permanent configuration allows you to control whether a custom property should be persisted and stored as part of the player's data or not. This can be useful for temporary properties, session-specific information, or data that doesn't need to be preserved beyond the current session.

For instance, you might use the non-permanent property to temporarily store a player's chat message before sending it to other players or logging it for debugging purposes. Since the property is not saved permanently, it won't clutter the saved data with transient information.

Here's an example of adding a new property to an item schema and extending the player schema:

typescript
import { RpgPlayer, RpgPlayerHooks } from '@rpgjs/server';

const itemSchemas = {
    ...RpgPlayer.schemas.items[0].item, // Keep existing properties synchronized
    graphic: String // Add a new property
};

export const player: RpgPlayerHooks = {
    props: {
        items: [{ nb: Number, item: itemSchemas }],
        equipments: [itemSchemas]
    }
};