@jsprismarine/prismarine
Version:
Dedicated Minecraft Bedrock Edition server written in TypeScript
326 lines (325 loc) • 41 kB
JavaScript
import ChatEvent from "./events/chat/ChatEvent.es.js";
import Timer from "./utils/Timer.es.js";
import TextType from "./network/type/TextType.es.js";
import ChangeDimensionPacket from "./network/packet/ChangeDimensionPacket.es.js";
import MovementType from "./network/type/MovementType.es.js";
import Human from "./entity/Human.es.js";
import { Chat, ChatType } from "./chat/Chat.es.js";
import PlayerSetGamemodeEvent from "./events/player/PlayerSetGamemodeEvent.es.js";
import PlayerToggleFlightEvent from "./events/player/PlayerToggleFlightEvent.es.js";
import PlayerToggleSprintEvent from "./events/player/PlayerToggleSprintEvent.es.js";
import PlayStatusType from "./network/type/PlayStatusType.es.js";
import CoordinateUtils from "./world/CoordinateUtils.es.js";
import PlayerSession from "./network/PlayerSession.es.js";
import { Vector3 } from "@jsprismarine/math";
import { Gametype, getGametypeId } from "@jsprismarine/minecraft";
//#region src/Player.ts
/**
* Default spawn view distance used in vanilla
*/
var VANILLA_DEFAULT_SPAWN_RADIUS = 4;
var Player = class extends Human {
address;
networkSession;
permissions;
/**
* Timer used for various metrics.
*/
timer;
connected = false;
xuid = "";
randomId = 0;
locale = "";
skin = null;
viewDistance = 0;
gamemode = Gametype.WORLD_DEFAULT;
onGround = false;
flying = false;
sneaking = false;
platformChatId = "";
device = null;
chunkSendQueue = /* @__PURE__ */ new Set();
/**
* Player's constructor.
*/
constructor({ connection, world, server, uuid }) {
super({
world,
server,
uuid
});
this.timer = new Timer();
this.address = connection.getRakNetSession().getAddress();
this.networkSession = new PlayerSession(server, connection, this);
this.permissions = [];
this.server.on("chat", this.chatHandler.bind(this));
}
/**
* On enable hook.
* @group Lifecycle
*/
async enable() {
const playerData = await this.getWorld().getPlayerData(this);
this.permissions = await this.server.getPermissionManager().getPermissions(this);
this.gamemode = getGametypeId(playerData.gamemode || this.server.getConfig().getGamemode());
this.setPosition({
position: playerData.position ? Vector3.fromObject(playerData.position) : await this.getWorld().getSpawnPosition(),
pitch: playerData.position?.pitch || 0,
yaw: playerData.position?.yaw || 0,
headYaw: playerData.position?.headYaw || 0,
type: MovementType.Reset
});
await this.sendPosition();
await this.sendSettings();
await Promise.all(this.getWorld().getPlayers().map((target) => this.getNetworkSession().broadcastMove(target, MovementType.Reset)));
this.server.getLogger().debug(`(Complete player creation took ${this.timer.stop()} ms)`);
this.connected = true;
}
/**
* On disable hook.
* @group Lifecycle
*/
async disable() {
if (this.connected && this.xuid) await this.getWorld().savePlayerData(this);
await this.getWorld().removeEntity(this);
await this.getNetworkSession().removeFromPlayerList();
for (const onlinePlayer of this.server.getSessionManager().getAllPlayers()) await this.getNetworkSession().sendDespawn(onlinePlayer);
const event = new ChatEvent(new Chat({
sender: this.server.getConsole(),
message: `§e%multiplayer.player.left`,
parameters: [this.getName()],
needsTranslation: true,
type: ChatType.TRANSLATION
}));
await this.server.emit("chat", event);
this.connected = false;
this.server.removeListener("chat", this.chatHandler);
}
async chatHandler(evt) {
if (evt.isCancelled()) return;
if (evt.getChat().getChannel() === "*.everyone" || evt.getChat().getChannel() === "*.ops" && this.isOp() || evt.getChat().getChannel() === `*.player.${this.getName()}`) await this.sendMessage(evt.getChat().getMessage(), evt.getChat().getType(), evt.getChat().getParameters(), evt.getChat().isNeedsTranslation());
}
/**
* Used to match vanilla behavior, will send chunks
* with an initial view radius of VANILLA_DEFAULT_SPAWN_RADIUS.
*/
async sendInitialSpawnChunks() {
const minX = CoordinateUtils.fromBlockToChunk(this.x) - 4;
const minZ = CoordinateUtils.fromBlockToChunk(this.z) - 4;
const maxX = CoordinateUtils.fromBlockToChunk(this.x) + 4;
const maxZ = CoordinateUtils.fromBlockToChunk(this.z) + 4;
const savedChunks = [];
const sendQueue = [];
for (let chunkX = minX; chunkX <= maxX; ++chunkX) for (let chunkZ = minZ; chunkZ <= maxZ; ++chunkZ) {
savedChunks.push({
x: chunkX,
z: chunkZ
});
sendQueue.push(this.getWorld().getChunk(chunkX, chunkZ));
}
await this.networkSession.sendNetworkChunkPublisher(4, savedChunks);
const getUniqueChunks = (sendList) => {
const xSet = /* @__PURE__ */ new Set();
const zSet = /* @__PURE__ */ new Set();
for (const coord of sendList) {
xSet.add(coord.x);
zSet.add(coord.z);
}
return Math.floor((xSet.size + zSet.size) / 2);
};
for (let i = 0; i < getUniqueChunks(savedChunks); ++i) await this.networkSession.sendNetworkChunkPublisher(4, []);
for await (const chunk of sendQueue) await this.networkSession.sendChunk(chunk);
}
/**
* Change the player's current world.
* @param {World} world - the new world
*/
async setWorld(world) {
const dim0 = new ChangeDimensionPacket();
dim0.dimension = 0;
dim0.position = this.getPosition();
dim0.respawn = false;
const dim1 = new ChangeDimensionPacket();
dim1.dimension = 1;
dim1.position = this.getPosition();
dim1.respawn = false;
await this.sendDespawn();
await this.getWorld().removeEntity(this);
await super.setWorld(world);
await world.addEntity(this);
await this.networkSession.send(dim0);
await this.networkSession.sendPlayStatus(PlayStatusType.PlayerSpawn);
await this.networkSession.send(dim1);
await this.networkSession.sendPlayStatus(PlayStatusType.PlayerSpawn);
await this.sendInitialSpawnChunks();
await this.networkSession.send(dim1);
await this.networkSession.sendPlayStatus(PlayStatusType.PlayerSpawn);
await this.networkSession.send(dim0);
await this.networkSession.sendPlayStatus(PlayStatusType.PlayerSpawn);
await this.networkSession.clearChunks();
await this.networkSession.needNewChunks();
await this.networkSession.sendPlayStatus(PlayStatusType.PlayerSpawn);
}
isOnline() {
return this.connected;
}
async update(tick) {
await super.update(tick);
await this.networkSession.update(tick);
if (tick % (6e3 * 1) === 0) await this.networkSession.sendTime(tick);
for (const chunk of this.chunkSendQueue) {
await this.networkSession.sendNetworkChunkPublisher(this.viewDistance || 4, []);
await this.networkSession.sendChunk(chunk);
this.chunkSendQueue.delete(chunk);
}
}
async kick(reason = "unknown reason") {
await this.disable();
await this.networkSession.kick(reason);
this.server.getLogger().verbose(`Player with id §b${this.getRuntimeId()}§r was kicked: ${reason}`);
}
async sendSettings() {
await this.getNetworkSession().sendGamemode();
await Promise.all(this.server.getSessionManager().getAllPlayers().map(async (target) => {
await target.getNetworkSession().sendAbilities(this);
await target.getNetworkSession().sendSettings(this);
}));
}
/**
* Player spawning logic.
*/
async sendSpawn() {
await this.sendPosition();
await this.setGamemode();
await this.getNetworkSession().sendInventory();
await this.sendSettings();
}
async sendDespawn() {}
/**
* Send a chat message to the client.
* @param message - the message
*/
async sendMessage(message, type = TextType.Raw, parameters = [], needsTranslation = false) {
await this.networkSession.sendMessage({
message,
parameters,
needsTranslation,
type
});
}
async setGamemode(mode) {
mode ??= this.gamemode;
const event = new PlayerSetGamemodeEvent(this, mode);
this.server.post(["playerSetGamemode", event]);
if (event.isCancelled()) return;
this.gamemode = event.getGamemode();
await this.networkSession.sendGamemode();
if (this.gamemode === Gametype.CREATIVE || this.gamemode === Gametype.SPECTATOR) this.metadata.setCanFly(true);
else {
this.metadata.setCanFly(false);
await this.setFlying(false);
}
await this.sendSettings();
await this.networkSession.sendGamemode();
await this.networkSession.sendMetadata();
}
getNetworkSession() {
return this.networkSession;
}
getAddress() {
return this.address;
}
getName() {
return this.metadata.nameTag;
}
getFormattedUsername() {
return `<${this.getName()}>`;
}
getPermissions() {
return this.permissions;
}
/**
* Get the XUID of the player.
* @returns {string | null} XUID of the player or null if not available
*/
getXUID() {
return this.xuid || null;
}
isPlayer() {
return true;
}
/**
* Check if the `Player` is an operator.
* @returns `true` if this player is an operator otherwise `false`.
*/
isOp() {
return this.server.getPermissionManager().isOp(this.getName());
}
async setSprinting(sprinting) {
if (sprinting === this.metadata.sprinting) return;
const event = new PlayerToggleSprintEvent(this, sprinting);
this.server.post(["playerToggleSprint", event]);
if (event.isCancelled()) return;
this.metadata.setSprinting(event.getIsSprinting());
await this.networkSession.sendMetadata();
}
isFlying() {
return this.flying;
}
async setFlying(flying) {
if (flying === this.isFlying()) return;
if (!this.metadata.canFly) {
this.flying = false;
await this.sendSettings();
return;
}
const event = new PlayerToggleFlightEvent(this, flying);
this.server.post(["playerToggleFlight", event]);
if (event.isCancelled()) return;
this.flying = event.getIsFlying();
await this.sendSettings();
}
isSneaking() {
return this.sneaking;
}
async setSneaking(val) {
if (val === this.sneaking) return;
this.sneaking = val;
}
isOnGround() {
return this.onGround;
}
async setOnGround(val) {
if (val === this.onGround) return;
this.onGround = val;
}
/**
* Set the position.
* @param {object} options - The options to set the position.
* @param {Vector3} options.position - The new position.
* @param {MovementType} [options.type=MovementType.Normal] - The movement type.
* @param {number} [options.pitch=this.pitch] - The new pitch.
* @param {number} [options.yaw=this.yaw] - The new yaw.
* @param {number} [options.headYaw=this.headYaw] - The new head yaw.
* @param {boolean} [broadcast=true] - Whether to broadcast the position change.
* @remarks This will notify the player's client about the position change.
*/
async setPosition({ position, type = MovementType.Normal, pitch = this.pitch, yaw = this.yaw, headYaw = this.headYaw }, broadcast = true) {
await super.setPosition({ position });
this.pitch = pitch;
this.yaw = yaw;
this.headYaw = headYaw;
if (!broadcast) return;
await this.networkSession.broadcastMove(this, type);
}
/**
* Send the position to all the players in the same world.
* @returns {Promise<void>} A promise that resolves when the position is sent.
*/
async sendPosition() {
await super.sendPosition();
}
};
//#endregion
export { VANILLA_DEFAULT_SPAWN_RADIUS, Player as default };
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"Player.es.js","names":[],"sources":["../src/Player.ts"],"sourcesContent":["import { Vector3 } from '@jsprismarine/math';\nimport { Gametype, getGametypeId } from '@jsprismarine/minecraft';\nimport type { InetAddress } from '@jsprismarine/raknet';\nimport type Server from './Server';\nimport { Chat, ChatType } from './chat/Chat';\nimport Human from './entity/Human';\nimport ChatEvent from './events/chat/ChatEvent';\nimport PlayerSetGamemodeEvent from './events/player/PlayerSetGamemodeEvent';\nimport PlayerToggleFlightEvent from './events/player/PlayerToggleFlightEvent';\nimport PlayerToggleSprintEvent from './events/player/PlayerToggleSprintEvent';\nimport type ClientConnection from './network/ClientConnection';\nimport { ChangeDimensionPacket } from './network/Packets';\nimport PlayerSession from './network/PlayerSession';\nimport type { ChunkCoord } from './network/packet/NetworkChunkPublisherUpdatePacket';\nimport MovementType from './network/type/MovementType';\nimport PlayStatusType from './network/type/PlayStatusType';\nimport TextType from './network/type/TextType';\nimport type Device from './utils/Device';\nimport Timer from './utils/Timer';\nimport type Skin from './utils/skin/Skin';\nimport type { World } from './world/';\nimport CoordinateUtils from './world/CoordinateUtils';\nimport type Chunk from './world/chunk/Chunk';\n\n/**\n * Default spawn view distance used in vanilla\n */\nexport const VANILLA_DEFAULT_SPAWN_RADIUS = 4;\n\nexport default class Player extends Human {\n    private readonly address: InetAddress;\n    private readonly networkSession: PlayerSession;\n    private permissions: string[];\n\n    /**\n     * Timer used for various metrics.\n     */\n    private timer: Timer;\n    private connected = false;\n\n    public xuid = '';\n    public randomId = 0;\n\n    public locale = '';\n    public skin: Skin | null = null;\n\n    public viewDistance = 0;\n    public gamemode: Gametype = Gametype.WORLD_DEFAULT;\n\n    private onGround = false;\n    private flying = false;\n    private sneaking = false;\n\n    public platformChatId = ''; // TODO: read this value from Login\n    public device: Device | null = null;\n\n    public readonly chunkSendQueue = new Set<Chunk>();\n\n    /**\n     * Player's constructor.\n     */\n    public constructor({\n        connection,\n        world,\n        server,\n        uuid\n    }: {\n        connection: ClientConnection;\n        world: World;\n        server: Server;\n        uuid?: string;\n    }) {\n        super({\n            world,\n            server,\n            uuid\n        });\n\n        this.timer = new Timer();\n\n        this.address = connection.getRakNetSession().getAddress();\n        this.networkSession = new PlayerSession(server, connection, this);\n        this.permissions = [];\n\n        this.server.on('chat', this.chatHandler.bind(this));\n    }\n\n    /**\n     * On enable hook.\n     * @group Lifecycle\n     */\n    public async enable(): Promise<void> {\n        const playerData = await this.getWorld().getPlayerData(this);\n\n        this.permissions = await this.server.getPermissionManager().getPermissions(this);\n        this.gamemode = getGametypeId(playerData.gamemode || this.server.getConfig().getGamemode());\n\n        this.setPosition({\n            position: playerData.position\n                ? Vector3.fromObject(playerData.position)\n                : await this.getWorld().getSpawnPosition(),\n            pitch: playerData.position?.pitch || 0,\n            yaw: playerData.position?.yaw || 0,\n            headYaw: playerData.position?.headYaw || 0,\n            type: MovementType.Reset\n        });\n        await this.sendPosition();\n\n        await this.sendSettings();\n\n        // Update position of all the players in the same world.\n        await Promise.all(\n            this.getWorld()\n                .getPlayers()\n                .map((target) => this.getNetworkSession().broadcastMove(target, MovementType.Reset))\n        );\n\n        // Finally mark the player as connected.\n        this.server.getLogger().debug(`(Complete player creation took ${this.timer.stop()} ms)`);\n        this.connected = true;\n    }\n\n    /**\n     * On disable hook.\n     * @group Lifecycle\n     */\n    public async disable(): Promise<void> {\n        if (this.connected && this.xuid) await this.getWorld().savePlayerData(this);\n        await this.getWorld().removeEntity(this);\n\n        // De-spawn the player to all online players\n        await this.getNetworkSession().removeFromPlayerList();\n        for (const onlinePlayer of this.server.getSessionManager().getAllPlayers()) {\n            await this.getNetworkSession().sendDespawn(onlinePlayer);\n        }\n\n        // Announce disconnection\n        const event = new ChatEvent(\n            new Chat({\n                sender: this.server.getConsole()!,\n                message: `§e%multiplayer.player.left`,\n                parameters: [this.getName()],\n                needsTranslation: true,\n                type: ChatType.TRANSLATION\n            })\n        );\n        await this.server.emit('chat', event);\n\n        this.connected = false;\n        this.server.removeListener('chat', this.chatHandler);\n    }\n\n    private async chatHandler(evt: ChatEvent) {\n        if (evt.isCancelled()) return;\n\n        // TODO: proper channel system\n        if (\n            evt.getChat().getChannel() === '*.everyone' ||\n            (evt.getChat().getChannel() === '*.ops' && this.isOp()) ||\n            evt.getChat().getChannel() === `*.player.${this.getName()}`\n        )\n            await this.sendMessage(\n                evt.getChat().getMessage(),\n                evt.getChat().getType() as number as TextType,\n                evt.getChat().getParameters(),\n                evt.getChat().isNeedsTranslation()\n            );\n    }\n\n    /**\n     * Used to match vanilla behavior, will send chunks\n     * with an initial view radius of VANILLA_DEFAULT_SPAWN_RADIUS.\n     */\n    public async sendInitialSpawnChunks(): Promise<void> {\n        const minX = CoordinateUtils.fromBlockToChunk(this.x) - VANILLA_DEFAULT_SPAWN_RADIUS;\n        const minZ = CoordinateUtils.fromBlockToChunk(this.z) - VANILLA_DEFAULT_SPAWN_RADIUS;\n        const maxX = CoordinateUtils.fromBlockToChunk(this.x) + VANILLA_DEFAULT_SPAWN_RADIUS;\n        const maxZ = CoordinateUtils.fromBlockToChunk(this.z) + VANILLA_DEFAULT_SPAWN_RADIUS;\n\n        const savedChunks: ChunkCoord[] = [];\n        const sendQueue: Array<Promise<Chunk>> = [];\n        for (let chunkX = minX; chunkX <= maxX; ++chunkX) {\n            for (let chunkZ = minZ; chunkZ <= maxZ; ++chunkZ) {\n                // TODO: vanilla does not send all of them, but in a range\n                // for example it does send them from x => [-3; 3] and z => [-3; 2]\n                savedChunks.push({ x: chunkX, z: chunkZ });\n                sendQueue.push(this.getWorld().getChunk(chunkX, chunkZ));\n            }\n        }\n\n        await this.networkSession.sendNetworkChunkPublisher(VANILLA_DEFAULT_SPAWN_RADIUS, savedChunks);\n\n        const getUniqueChunks = (sendList: ChunkCoord[]) => {\n            const xSet = new Set<number>();\n            const zSet = new Set<number>();\n\n            for (const coord of sendList) {\n                xSet.add(coord.x);\n                zSet.add(coord.z);\n            }\n\n            return Math.floor((xSet.size + zSet.size) / 2);\n        };\n\n        for (let i = 0; i < getUniqueChunks(savedChunks); ++i) {\n            await this.networkSession.sendNetworkChunkPublisher(VANILLA_DEFAULT_SPAWN_RADIUS, []);\n        }\n\n        for await (const chunk of sendQueue) {\n            await this.networkSession.sendChunk(chunk);\n        }\n    }\n\n    /**\n     * Change the player's current world.\n     * @param {World} world - the new world\n     */\n    public async setWorld(world: World) {\n        const dim0 = new ChangeDimensionPacket();\n        dim0.dimension = 0;\n        dim0.position = this.getPosition();\n        dim0.respawn = false;\n\n        const dim1 = new ChangeDimensionPacket();\n        dim1.dimension = 1;\n        dim1.position = this.getPosition();\n        dim1.respawn = false;\n\n        await this.sendDespawn();\n        await this.getWorld().removeEntity(this);\n\n        await super.setWorld(world);\n        await world.addEntity(this);\n\n        await this.networkSession.send(dim0);\n        await this.networkSession.sendPlayStatus(PlayStatusType.PlayerSpawn);\n        await this.networkSession.send(dim1);\n        await this.networkSession.sendPlayStatus(PlayStatusType.PlayerSpawn);\n\n        await this.sendInitialSpawnChunks();\n\n        await this.networkSession.send(dim1);\n        await this.networkSession.sendPlayStatus(PlayStatusType.PlayerSpawn);\n        await this.networkSession.send(dim0);\n        await this.networkSession.sendPlayStatus(PlayStatusType.PlayerSpawn);\n\n        await this.networkSession.clearChunks();\n        await this.networkSession.needNewChunks();\n        await this.networkSession.sendPlayStatus(PlayStatusType.PlayerSpawn);\n    }\n\n    public isOnline() {\n        return this.connected;\n    }\n\n    public async update(tick: number): Promise<void> {\n        await super.update(tick);\n        await this.networkSession.update(tick);\n\n        // TODO: get documentation about timings from vanilla\n        // 1 second / 20 = 1 tick, 20 * 5 = 1 second\n        // 1 second * 60 = 1 minute\n        if (tick % (20 * 5 * 60 * 1) === 0) {\n            await this.networkSession.sendTime(tick);\n        }\n\n        for (const chunk of this.chunkSendQueue) {\n            await this.networkSession.sendNetworkChunkPublisher(this.viewDistance || VANILLA_DEFAULT_SPAWN_RADIUS, []);\n            await this.networkSession.sendChunk(chunk);\n            this.chunkSendQueue.delete(chunk);\n        }\n    }\n\n    public async kick(reason = 'unknown reason'): Promise<void> {\n        await this.disable();\n        await this.networkSession.kick(reason);\n        this.server.getLogger().verbose(`Player with id §b${this.getRuntimeId()}§r was kicked: ${reason}`);\n    }\n\n    public async sendSettings(): Promise<void> {\n        await this.getNetworkSession().sendGamemode();\n\n        await Promise.all(\n            this.server\n                .getSessionManager()\n                .getAllPlayers()\n                .map(async (target) => {\n                    await target.getNetworkSession().sendAbilities(this);\n                    await target.getNetworkSession().sendSettings(this);\n                })\n        );\n    }\n\n    /**\n     * Player spawning logic.\n     */\n    public async sendSpawn() {\n        await this.sendPosition();\n        await this.setGamemode();\n        await this.getNetworkSession().sendInventory();\n        await this.sendSettings();\n    }\n    public async sendDespawn() {}\n\n    /**\n     * Send a chat message to the client.\n     * @param message - the message\n     */\n    public async sendMessage(\n        message: string,\n        type: TextType = TextType.Raw,\n        parameters: string[] = [],\n        needsTranslation = false\n    ): Promise<void> {\n        // TODO: Do this properly like java edition,\n        // in other words, the message should be JSON formatted.\n        await this.networkSession.sendMessage({\n            message,\n            parameters,\n            needsTranslation,\n            type\n        });\n    }\n\n    public async setGamemode(mode?: Gametype): Promise<void> {\n        mode ??= this.gamemode;\n\n        const event = new PlayerSetGamemodeEvent(this, mode);\n        this.server.post(['playerSetGamemode', event]);\n        if (event.isCancelled()) return;\n\n        this.gamemode = event.getGamemode();\n        await this.networkSession.sendGamemode();\n\n        if (this.gamemode === Gametype.CREATIVE || this.gamemode === Gametype.SPECTATOR) {\n            this.metadata.setCanFly(true);\n        } else {\n            this.metadata.setCanFly(false);\n            await this.setFlying(false);\n        }\n\n        await this.sendSettings();\n        await this.networkSession.sendGamemode();\n        await this.networkSession.sendMetadata();\n    }\n\n    public getNetworkSession(): PlayerSession {\n        return this.networkSession;\n    }\n\n    public getAddress() {\n        return this.address;\n    }\n\n    public getName(): string {\n        return this.metadata.nameTag;\n    }\n\n    public getFormattedUsername(): string {\n        return `<${this.getName()}>`;\n    }\n\n    public getPermissions(): string[] {\n        return this.permissions;\n    }\n\n    /**\n     * Get the XUID of the player.\n     * @returns {string | null} XUID of the player or null if not available\n     */\n    public getXUID(): string | null {\n        return this.xuid || null;\n    }\n\n    public isPlayer(): boolean {\n        return true;\n    }\n\n    /**\n     * Check if the `Player` is an operator.\n     * @returns `true` if this player is an operator otherwise `false`.\n     */\n    public isOp(): boolean {\n        return this.server.getPermissionManager().isOp(this.getName());\n    }\n\n    public async setSprinting(sprinting: boolean) {\n        if (sprinting === this.metadata.sprinting) return;\n\n        const event = new PlayerToggleSprintEvent(this, sprinting);\n        this.server.post(['playerToggleSprint', event]);\n        if (event.isCancelled()) return;\n\n        this.metadata.setSprinting(event.getIsSprinting());\n        await this.networkSession.sendMetadata();\n    }\n\n    public isFlying() {\n        return this.flying;\n    }\n    public async setFlying(flying: boolean) {\n        if (flying === this.isFlying()) return;\n\n        if (!this.metadata.canFly) {\n            this.flying = false;\n            await this.sendSettings();\n            return;\n        }\n\n        const event = new PlayerToggleFlightEvent(this, flying);\n        this.server.post(['playerToggleFlight', event]);\n        if (event.isCancelled()) return;\n\n        this.flying = event.getIsFlying();\n        await this.sendSettings();\n    }\n\n    public isSneaking() {\n        return this.sneaking;\n    }\n    public async setSneaking(val: boolean) {\n        if (val === this.sneaking) return;\n        this.sneaking = val;\n    }\n\n    public isOnGround() {\n        return this.onGround;\n    }\n    public async setOnGround(val: boolean) {\n        if (val === this.onGround) return;\n        this.onGround = val;\n    }\n\n    /**\n     * Set the position.\n     * @param {object} options - The options to set the position.\n     * @param {Vector3} options.position - The new position.\n     * @param {MovementType} [options.type=MovementType.Normal] - The movement type.\n     * @param {number} [options.pitch=this.pitch] - The new pitch.\n     * @param {number} [options.yaw=this.yaw] - The new yaw.\n     * @param {number} [options.headYaw=this.headYaw] - The new head yaw.\n     * @param {boolean} [broadcast=true] - Whether to broadcast the position change.\n     * @remarks This will notify the player's client about the position change.\n     */\n    public async setPosition(\n        {\n            position,\n            type = MovementType.Normal,\n            pitch = this.pitch,\n            yaw = this.yaw,\n            headYaw = this.headYaw\n        }: {\n            position: Vector3;\n            type?: MovementType;\n            pitch?: number;\n            yaw?: number;\n            headYaw?: number;\n        },\n        broadcast = true\n    ) {\n        await super.setPosition({ position });\n        this.pitch = pitch;\n        this.yaw = yaw;\n        this.headYaw = headYaw;\n\n        if (!broadcast) return;\n        await this.networkSession.broadcastMove(this, type);\n    }\n    /**\n     * Send the position to all the players in the same world.\n     * @returns {Promise<void>} A promise that resolves when the position is sent.\n     */\n    public async sendPosition(): Promise<void> {\n        await super.sendPosition();\n    }\n}\n"],"mappings":"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"}