UNPKG

@jsprismarine/prismarine

Version:

Dedicated Minecraft Bedrock Edition server written in TypeScript

326 lines (325 loc) 41 kB
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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"}