From 691abb7c4f590d61dd7a385cd2da62b4b5c881c9 Mon Sep 17 00:00:00 2001 From: Dennis Postma Date: Thu, 26 Dec 2024 15:15:18 +0100 Subject: [PATCH] Added getter & setter functions to all entities --- src/application/bases/baseEntity.ts | 2 +- src/application/types.ts | 2 + src/commands/init.ts | 251 +++++++++++++------------- src/entities/character.ts | 207 +++++++++++++++++++++ src/entities/characterEquipment.ts | 36 ++++ src/entities/characterHair.ts | 54 ++++++ src/entities/characterItem.ts | 45 +++++ src/entities/characterType.ts | 81 +++++++++ src/entities/chat.ts | 45 +++++ src/entities/item.ts | 94 +++++++++- src/entities/mapObject.ts | 109 +++++++++++ src/entities/passwordResetToken.ts | 36 ++++ src/entities/sprite.ts | 37 ++++ src/entities/spriteAction.ts | 102 ++++++++++- src/entities/tile.ts | 46 +++++ src/entities/user.ts | 65 +++++++ src/entities/world.ts | 45 +++++ src/entities/zone.ts | 126 +++++++++++++ src/entities/zoneEffect.ts | 37 ++++ src/entities/zoneEventTile.ts | 58 +++++- src/entities/zoneEventTileTeleport.ts | 55 ++++++ src/entities/zoneObject.ts | 67 ++++++- src/http/avatar.ts | 8 +- src/managers/zoneManager.ts | 5 +- src/models/zoneCharacter.ts | 2 +- src/repositories/worldRepository.ts | 2 +- src/repositories/zoneRepository.ts | 2 +- src/services/characterService.ts | 2 +- src/services/chatService.ts | 8 +- src/services/userService.ts | 12 +- src/services/worldService.ts | 10 +- src/services/zoneEventTileService.ts | 15 +- src/socketEvents/disconnect.ts | 2 +- 33 files changed, 1500 insertions(+), 168 deletions(-) diff --git a/src/application/bases/baseEntity.ts b/src/application/bases/baseEntity.ts index 0befbd2..c135cc3 100644 --- a/src/application/bases/baseEntity.ts +++ b/src/application/bases/baseEntity.ts @@ -64,4 +64,4 @@ export abstract class BaseEntity { throw error } } -} \ No newline at end of file +} diff --git a/src/application/types.ts b/src/application/types.ts index 0df12d9..d086b95 100644 --- a/src/application/types.ts +++ b/src/application/types.ts @@ -3,6 +3,8 @@ import { Character } from '#entities/character' import { ZoneEventTile } from '#entities/zoneEventTile' import { ZoneEventTileTeleport } from '#entities/zoneEventTileTeleport' +export type UUID = `${string}-${string}-${string}-${string}-${string}` + export type TSocket = Socket & { userId?: number characterId?: number diff --git a/src/commands/init.ts b/src/commands/init.ts index 1d1600d..673e6d2 100644 --- a/src/commands/init.ts +++ b/src/commands/init.ts @@ -2,7 +2,6 @@ import { Server } from 'socket.io' import fs from 'fs' import { getPublicPath } from '#application/storage' import sharp from 'sharp' -import bcrypt from 'bcryptjs' import { Tile } from '#entities/tile' import { MapObject } from '#entities/mapObject' import { SpriteAction } from '#entities/spriteAction' @@ -17,6 +16,7 @@ import ZoneRepository from '#repositories/zoneRepository' import { CharacterType } from '#entities/characterType' import { CharacterGender, CharacterRace } from '#application/enums' import { CharacterHair } from '#entities/characterHair' +import { UUID } from '#application/types' // @TODO : Replace this with seeding // https://mikro-orm.io/docs/seeding @@ -45,8 +45,8 @@ export default class InitCommand { private async importTiles(): Promise { for (const tile of fs.readdirSync(getPublicPath('tiles'))) { const newTile = new Tile() - newTile.id = tile.split('.')[0] as any - newTile.name = 'New tile' + newTile.setId(tile.split('.')[0] as UUID).setName('New tile') + await newTile.save() } } @@ -54,16 +54,19 @@ export default class InitCommand { private async importObjects(): Promise { for (const object of fs.readdirSync(getPublicPath('objects'))) { const newMapObject = new MapObject() - newMapObject.id = object.split('.')[0] as any - newMapObject.name = 'New object' - - newMapObject.frameWidth = await sharp(getPublicPath('objects', object)) - .metadata() - .then((metadata) => metadata.height) ?? 0 - - newMapObject.frameHeight = await sharp(getPublicPath('objects', object)) - .metadata() - .then((metadata) => metadata.width) ?? 0 + newMapObject + .setId(object.split('.')[0] as UUID) + .setName('New object') + .setFrameWidth( + (await sharp(getPublicPath('objects', object)) + .metadata() + .then((metadata) => metadata.height)) ?? 0 + ) + .setFrameHeight( + (await sharp(getPublicPath('objects', object)) + .metadata() + .then((metadata) => metadata.width)) ?? 0 + ) await newMapObject.save() } @@ -71,129 +74,123 @@ export default class InitCommand { private async createCharacterType(): Promise { const characterSprite = new Sprite() - characterSprite.id = '023d1e9d-f57f-4faa-8412-86c07107cf85' - characterSprite.name = 'Character' + characterSprite.setId('023d1e9d-f57f-4faa-8412-86c07107cf85').setName('Character') await characterSprite.save() const idleRightDownAction = new SpriteAction() - idleRightDownAction.action = 'idle_right_down' - idleRightDownAction.sprites = [ - "" - ] // Base64 sprite data - idleRightDownAction.originX = 0 - idleRightDownAction.originY = 0 - idleRightDownAction.isAnimated = false - idleRightDownAction.isLooping = false - idleRightDownAction.frameWidth = 64 - idleRightDownAction.frameHeight = 94 - idleRightDownAction.frameRate = 0 - idleRightDownAction.sprite = characterSprite + idleRightDownAction + .setAction('idle_right_down') + .setSprites([ + '' + ]) + .setOriginX(0) + .setOriginY(0) + .setIsAnimated(false) + .setIsLooping(false) + .setFrameWidth(64) + .setFrameHeight(94) + .setFrameRate(0) + .setSprite(characterSprite) await idleRightDownAction.save() const idleLeftUpAction = new SpriteAction() - idleLeftUpAction.action = 'idle_left_up' - idleLeftUpAction.sprites = [ - "" - ] // Base64 sprite data - idleLeftUpAction.originX = 0 - idleLeftUpAction.originY = 0 - idleLeftUpAction.isAnimated = false - idleLeftUpAction.isLooping = false - idleLeftUpAction.frameWidth = 64 - idleLeftUpAction.frameHeight = 94 - idleLeftUpAction.frameRate = 0 - idleLeftUpAction.sprite = characterSprite + idleLeftUpAction + .setAction('idle_left_up') + .setSprites([ + '' + ]) + .setOriginX(0) + .setOriginY(0) + .setIsAnimated(false) + .setIsLooping(false) + .setFrameWidth(64) + .setFrameHeight(94) + .setFrameRate(0) + .setSprite(characterSprite) await idleLeftUpAction.save() const walkRightDownAction = new SpriteAction() - - walkRightDownAction.action = 'walk_right_down' - walkRightDownAction.sprites = [ - "", - "", - "", - "" - ] // Base64 sprite data array - walkRightDownAction.originX = 0 - walkRightDownAction.originY = 0 - walkRightDownAction.isAnimated = true - walkRightDownAction.isLooping = false - walkRightDownAction.frameWidth = 64 - walkRightDownAction.frameHeight = 94 - walkRightDownAction.frameRate = 7 - walkRightDownAction.sprite = characterSprite + walkRightDownAction + .setAction('walk_right_down') + .setSprites([ + '', + '', + '', + '' + ]) + .setOriginX(0) + .setOriginY(0) + .setIsAnimated(true) + .setIsLooping(false) + .setFrameWidth(64) + .setFrameHeight(94) + .setFrameRate(7) + .setSprite(characterSprite) await walkRightDownAction.save() const walkLeftUpAction = new SpriteAction() - walkLeftUpAction.action = 'walk_left_up' - walkLeftUpAction.sprites = [ - "", - "", - "", - "" - ] // Base64 sprite data array - walkLeftUpAction.originX = 0 - walkLeftUpAction.originY = 0 - walkLeftUpAction.isAnimated = true - walkLeftUpAction.isLooping = false - walkLeftUpAction.frameWidth = 64 - walkLeftUpAction.frameHeight = 94 - walkLeftUpAction.frameRate = 7 - walkLeftUpAction.sprite = characterSprite + walkLeftUpAction + .setAction('walk_left_up') + .setSprites([ + '', + '', + '', + '' + ]) + .setOriginX(0) + .setOriginY(0) + .setIsAnimated(true) + .setIsLooping(false) + .setFrameWidth(64) + .setFrameHeight(94) + .setFrameRate(7) + .setSprite(characterSprite) await walkLeftUpAction.save() const characterType = new CharacterType() - characterType.id = 1 - characterType.name = 'New character type' - characterType.gender = CharacterGender.MALE - characterType.race = CharacterRace.HUMAN - characterType.isSelectable = true - characterType.sprite = characterSprite + characterType.setId(1).setName('New character type').setGender(CharacterGender.MALE).setRace(CharacterRace.HUMAN).setIsSelectable(true).setSprite(characterSprite) await characterType.save() } private async createCharacterHair(): Promise { const hairSprite = new Sprite() - hairSprite.id = '922ee95f-1500-49c0-8ead-f8cc46dad136' - hairSprite.name = 'Hair 1' + hairSprite.setId('922ee95f-1500-49c0-8ead-f8cc46dad136').setName('Hair 1') await hairSprite.save() const frontAction = new SpriteAction() - frontAction.action = 'front' - frontAction.sprites = [ - "" - ] // Base64 sprite data - frontAction.originX = 0.5 - frontAction.originY = 5.34 - frontAction.isAnimated = false - frontAction.isLooping = false - frontAction.frameWidth = 64 - frontAction.frameHeight = 18 - frontAction.frameRate = 0 - frontAction.sprite = hairSprite + frontAction + .setAction('front') + .setSprites([ + '' + ]) + .setOriginX(0.5) + .setOriginY(5.34) + .setIsAnimated(false) + .setIsLooping(false) + .setFrameWidth(64) + .setFrameHeight(18) + .setFrameRate(0) + .setSprite(hairSprite) await frontAction.save() const backAction = new SpriteAction() - backAction.action = 'back' - backAction.sprites = [ - "" - ] // Base64 sprite data - backAction.originX = 0.5 - backAction.originY = 4.34 - backAction.isAnimated = false - backAction.isLooping = false - backAction.frameWidth = 64 - backAction.frameHeight = 22 - backAction.frameRate = 0 - backAction.sprite = hairSprite + backAction + .setAction('back') + .setSprites([ + '' + ]) + .setOriginX(0.5) + .setOriginY(4.34) + .setIsAnimated(false) + .setIsLooping(false) + .setFrameWidth(64) + .setFrameHeight(22) + .setFrameRate(0) + .setSprite(hairSprite) await backAction.save() const characterHair = new CharacterHair() - characterHair.id = 1 - characterHair.name = 'Hair 1' - characterHair.gender = CharacterGender.MALE - characterHair.isSelectable = true - characterHair.sprite = hairSprite + characterHair.setId(1).setName('Hair 1').setGender(CharacterGender.MALE).setIsSelectable(true).setSprite(hairSprite) await characterHair.save() } @@ -215,7 +212,7 @@ export default class InitCommand { frameWidth: 64, frameHeight: 94, frameRate: 0 - }, + } // Add other actions... ] @@ -231,38 +228,32 @@ export default class InitCommand { private async createZone(): Promise { const zone = new Zone() - zone.name = 'New zone' - zone.width = 100 - zone.height = 100 - zone.tiles = Array.from({ length: 100 }, () => - Array.from({ length: 100 }, () => 'a2fd8d6f-5042-437a-9c1e-c66b91ecc35b') - ) + zone + .setName('New zone') + .setWidth(100) + .setHeight(100) + .setTiles(Array.from({ length: 100 }, () => Array.from({ length: 100 }, () => 'a2fd8d6f-5042-437a-9c1e-c66b91ecc35b'))) await zone.save() const effect = new ZoneEffect() - effect.effect = 'light' - effect.strength = 100 - effect.zone = zone + effect.setEffect('light').setStrength(100).setZone(zone) await effect.save() } private async createUser(): Promise { const user = new User() - user.id = 1 - user.username = 'root' - user.email = 'local@host' - user.password = await bcrypt.hash('password', 10) - user.online = false + user.setId(1).setUsername('root').setEmail('local@host').setPassword('password').setOnline(false) await user.save() const character = new Character() - character.id = 1 - character.user = user - character.name = 'root' - character.role = 'gm' - character.zone = await ZoneRepository.getFirst() ?? undefined - character.characterType = await CharacterTypeRepository.getFirst() ?? undefined - character.characterHair = await CharacterHairRepository.getFirst() ?? undefined + character + .setId(1) + .setUser(user) + .setName('root') + .setRole('gm') + .setZone((await ZoneRepository.getFirst()) ?? undefined) + .setCharacterType((await CharacterTypeRepository.getFirst()) ?? undefined) + .setCharacterHair((await CharacterHairRepository.getFirst()) ?? undefined) await character.save() } -} \ No newline at end of file +} diff --git a/src/entities/character.ts b/src/entities/character.ts index f108120..65e0c11 100644 --- a/src/entities/character.ts +++ b/src/entities/character.ts @@ -82,4 +82,211 @@ export class Character extends BaseEntity { @Property() wisdom = 10 + + setId(id: number) { + this.id = id + return this + } + + getId() { + return this.id + } + + setUser(user: User) { + this.user = user + return this + } + + getUser() { + return this.user + } + + setName(name: string) { + this.name = name + return this + } + + getName() { + return this.name + } + + setOnline(online: boolean) { + this.online = online + return this + } + + getOnline() { + return this.online + } + + setRole(role: string) { + this.role = role + return this + } + + getRole() { + return this.role + } + + setChats(chats: Collection) { + this.chats = chats + return this + } + + getChats() { + return this.chats + } + + setZone(zone: Zone | undefined) { + this.zone = zone + return this + } + + getZone() { + return this.zone + } + + setPositionX(positionX: number) { + this.positionX = positionX + return this + } + + getPositionX() { + return this.positionX + } + + setPositionY(positionY: number) { + this.positionY = positionY + return this + } + + getPositionY() { + return this.positionY + } + + setRotation(rotation: number) { + this.rotation = rotation + return this + } + + getRotation() { + return this.rotation + } + + setCharacterType(characterType: CharacterType | undefined) { + this.characterType = characterType + return this + } + + getCharacterType() { + return this.characterType + } + + setCharacterHair(characterHair: CharacterHair | undefined) { + this.characterHair = characterHair + return this + } + + getCharacterHair() { + return this.characterHair + } + + setItems(items: Collection) { + this.items = items + return this + } + + getItems() { + return this.items + } + + setEquipment(equipment: Collection) { + this.equipment = equipment + return this + } + + getEquipment() { + return this.equipment + } + + setAlignment(alignment: number) { + this.alignment = alignment + return this + } + + getAlignment() { + return this.alignment + } + + setHitpoints(hitpoints: number) { + this.hitpoints = hitpoints + return this + } + + getHitpoints() { + return this.hitpoints + } + + setMana(mana: number) { + this.mana = mana + return this + } + + getMana() { + return this.mana + } + + setLevel(level: number) { + this.level = level + return this + } + + getLevel() { + return this.level + } + + setExperience(experience: number) { + this.experience = experience + return this + } + + getExperience() { + return this.experience + } + + setStrength(strength: number) { + this.strength = strength + return this + } + + getStrength() { + return this.strength + } + + setDexterity(dexterity: number) { + this.dexterity = dexterity + return this + } + + getDexterity() { + return this.dexterity + } + + setIntelligence(intelligence: number) { + this.intelligence = intelligence + return this + } + + getIntelligence() { + return this.intelligence + } + + setWisdom(wisdom: number) { + this.wisdom = wisdom + return this + } + + getWisdom() { + return this.wisdom + } } diff --git a/src/entities/characterEquipment.ts b/src/entities/characterEquipment.ts index a82ef7a..f79d2a8 100644 --- a/src/entities/characterEquipment.ts +++ b/src/entities/characterEquipment.ts @@ -17,4 +17,40 @@ export class CharacterEquipment extends BaseEntity { @ManyToOne(() => CharacterItem) characterItem!: CharacterItem + + setId(id: number) { + this.id = id + return this + } + + getId() { + return this.id + } + + setSlot(slot: CharacterEquipmentSlotType) { + this.slot = slot + return this + } + + getSlot() { + return this.slot + } + + setCharacter(character: Character) { + this.character = character + return this + } + + getCharacter() { + return this.character + } + + setCharacterItem(characterItem: CharacterItem) { + this.characterItem = characterItem + return this + } + + getCharacterItem() { + return this.characterItem + } } diff --git a/src/entities/characterHair.ts b/src/entities/characterHair.ts index a5e66c3..871e12a 100644 --- a/src/entities/characterHair.ts +++ b/src/entities/characterHair.ts @@ -23,4 +23,58 @@ export class CharacterHair extends BaseEntity { @OneToMany(() => Character, (character) => character.characterHair) characters = new Collection(this) + + setId(id: number) { + this.id = id + return this + } + + getId() { + return this.id + } + + setName(name: string) { + this.name = name + return this + } + + getName() { + return this.name + } + + setGender(gender: CharacterGender) { + this.gender = gender + return this + } + + getGender() { + return this.gender + } + + setIsSelectable(isSelectable: boolean) { + this.isSelectable = isSelectable + return this + } + + getIsSelectable() { + return this.isSelectable + } + + setSprite(sprite: Sprite) { + this.sprite = sprite + return this + } + + getSprite() { + return this.sprite + } + + setCharacters(characters: Collection) { + this.characters = characters + return this + } + + getCharacters() { + return this.characters + } } diff --git a/src/entities/characterItem.ts b/src/entities/characterItem.ts index e008029..9b162c0 100644 --- a/src/entities/characterItem.ts +++ b/src/entities/characterItem.ts @@ -20,4 +20,49 @@ export class CharacterItem extends BaseEntity { @OneToMany(() => CharacterEquipment, (equipment) => equipment.characterItem) characterEquipment = new Collection(this) + + setId(id: number) { + this.id = id + return this + } + + getId() { + return this.id + } + + setCharacter(character: Character) { + this.character = character + return this + } + + getCharacter() { + return this.character + } + + setItem(item: Item) { + this.item = item + return this + } + + getItem() { + return this.item + } + + setQuantity(quantity: number) { + this.quantity = quantity + return this + } + + getQuantity() { + return this.quantity + } + + setCharacterEquipment(characterEquipment: Collection) { + this.characterEquipment = characterEquipment + return this + } + + getCharacterEquipment() { + return this.characterEquipment + } } diff --git a/src/entities/characterType.ts b/src/entities/characterType.ts index 8fd907a..7731233 100644 --- a/src/entities/characterType.ts +++ b/src/entities/characterType.ts @@ -32,4 +32,85 @@ export class CharacterType extends BaseEntity { @Property() updatedAt = new Date() + + setId(id: number) { + this.id = id + return this + } + + getId() { + return this.id + } + + setName(name: string) { + this.name = name + return this + } + + getName() { + return this.name + } + + setGender(gender: CharacterGender) { + this.gender = gender + return this + } + + getGender() { + return this.gender + } + + setRace(race: CharacterRace) { + this.race = race + return this + } + + getRace() { + return this.race + } + + setIsSelectable(isSelectable: boolean) { + this.isSelectable = isSelectable + return this + } + + getIsSelectable() { + return this.isSelectable + } + + setSprite(sprite: Sprite) { + this.sprite = sprite + return this + } + + getSprite() { + return this.sprite + } + + setCreatedAt(createdAt: Date) { + this.createdAt = createdAt + return this + } + + getCreatedAt() { + return this.createdAt + } + + setUpdatedAt(updatedAt: Date) { + this.updatedAt = updatedAt + return this + } + + getUpdatedAt() { + return this.updatedAt + } + + setCharacters(characters: Collection) { + this.characters = characters + return this + } + + getCharacters() { + return this.characters + } } diff --git a/src/entities/chat.ts b/src/entities/chat.ts index 8182425..2d232eb 100644 --- a/src/entities/chat.ts +++ b/src/entities/chat.ts @@ -19,4 +19,49 @@ export class Chat extends BaseEntity { @Property() createdAt = new Date() + + setId(id: number) { + this.id = id + return this + } + + getId() { + return this.id + } + + setCharacter(character: Character) { + this.character = character + return this + } + + getCharacter() { + return this.character + } + + setZone(zone: Zone) { + this.zone = zone + return this + } + + getZone() { + return this.zone + } + + setMessage(message: string) { + this.message = message + return this + } + + getMessage() { + return this.message + } + + setCreatedAt(createdAt: Date) { + this.createdAt = createdAt + return this + } + + getCreatedAt() { + return this.createdAt + } } diff --git a/src/entities/item.ts b/src/entities/item.ts index 9cc08e0..21f3fdb 100644 --- a/src/entities/item.ts +++ b/src/entities/item.ts @@ -3,11 +3,13 @@ import { BaseEntity } from '#application/bases/baseEntity' import { Sprite } from './sprite' import { CharacterItem } from './characterItem' import { ItemType, ItemRarity } from '#application/enums' +import { randomUUID } from 'node:crypto' +import { UUID } from '#application/types' @Entity() export class Item extends BaseEntity { @PrimaryKey() - id!: string + id = randomUUID() @Property() name!: string @@ -35,4 +37,94 @@ export class Item extends BaseEntity { @OneToMany(() => CharacterItem, (characterItem) => characterItem.item) characters = new Collection(this) + + setId(id: UUID) { + this.id = id + return this + } + + getId() { + return this.id + } + + setName(name: string) { + this.name = name + return this + } + + getName() { + return this.name + } + + setDescription(description: string) { + this.description = description + return this + } + + getDescription() { + return this.description + } + + setItemType(itemType: ItemType) { + this.itemType = itemType + return this + } + + getItemType() { + return this.itemType + } + + setStackable(stackable: boolean) { + this.stackable = stackable + return this + } + + getStackable() { + return this.stackable + } + + setRarity(rarity: ItemRarity) { + this.rarity = rarity + return this + } + + getRarity() { + return this.rarity + } + + setSprite(sprite: Sprite) { + this.sprite = sprite + return this + } + + getSprite() { + return this.sprite + } + + setCreatedAt(createdAt: Date) { + this.createdAt = createdAt + return this + } + + getCreatedAt() { + return this.createdAt + } + + setUpdatedAt(updatedAt: Date) { + this.updatedAt = updatedAt + return this + } + + getUpdatedAt() { + return this.updatedAt + } + + setCharacters(characters: Collection) { + this.characters = characters + return this + } + + getCharacters() { + return this.characters + } } diff --git a/src/entities/mapObject.ts b/src/entities/mapObject.ts index e1edd77..6b2879d 100644 --- a/src/entities/mapObject.ts +++ b/src/entities/mapObject.ts @@ -2,6 +2,7 @@ import { randomUUID } from 'node:crypto' import { Collection, Entity, OneToMany, PrimaryKey, Property } from '@mikro-orm/core' import { BaseEntity } from '#application/bases/baseEntity' import { ZoneObject } from './zoneObject' +import { UUID } from '#application/types' @Entity() export class MapObject extends BaseEntity { @@ -40,4 +41,112 @@ export class MapObject extends BaseEntity { @OneToMany(() => ZoneObject, (zoneObject) => zoneObject.mapObject) zoneObjects = new Collection(this) + + setId(id: UUID) { + this.id = id + return this + } + + getId() { + return this.id + } + + setName(name: string) { + this.name = name + return this + } + + getName() { + return this.name + } + + setTags(tags: any) { + this.tags = tags + return this + } + + getTags() { + return this.tags + } + + setOriginX(originX: number) { + this.originX = originX + return this + } + + getOriginX() { + return this.originX + } + + setOriginY(originY: number) { + this.originY = originY + return this + } + + getOriginY() { + return this.originY + } + + setIsAnimated(isAnimated: boolean) { + this.isAnimated = isAnimated + return this + } + + getIsAnimated() { + return this.isAnimated + } + + setFrameRate(frameRate: number) { + this.frameRate = frameRate + return this + } + + getFrameRate() { + return this.frameRate + } + + setFrameWidth(frameWidth: number) { + this.frameWidth = frameWidth + return this + } + + getFrameWidth() { + return this.frameWidth + } + + setFrameHeight(frameHeight: number) { + this.frameHeight = frameHeight + return this + } + + getFrameHeight() { + return this.frameHeight + } + + setCreatedAt(createdAt: Date) { + this.createdAt = createdAt + return this + } + + getCreatedAt() { + return this.createdAt + } + + setUpdatedAt(updatedAt: Date) { + this.updatedAt = updatedAt + return this + } + + getUpdatedAt() { + return this.updatedAt + } + + setZoneObjects(zoneObjects: Collection) { + this.zoneObjects = zoneObjects + return this + } + + getZoneObjects() { + return this.zoneObjects + } } diff --git a/src/entities/passwordResetToken.ts b/src/entities/passwordResetToken.ts index 96d215b..070ec2e 100644 --- a/src/entities/passwordResetToken.ts +++ b/src/entities/passwordResetToken.ts @@ -15,4 +15,40 @@ export class PasswordResetToken extends BaseEntity { @Property() createdAt = new Date() + + setId(id: number) { + this.id = id + return this + } + + getId() { + return this.id + } + + setUser(user: User) { + this.user = user + return this + } + + getUser() { + return this.user + } + + setToken(token: string) { + this.token = token + return this + } + + getToken() { + return this.token + } + + setCreatedAt(createdAt: Date) { + this.createdAt = createdAt + return this + } + + getCreatedAt() { + return this.createdAt + } } diff --git a/src/entities/sprite.ts b/src/entities/sprite.ts index 91fda82..b53f24c 100644 --- a/src/entities/sprite.ts +++ b/src/entities/sprite.ts @@ -5,6 +5,7 @@ import { SpriteAction } from './spriteAction' import { CharacterType } from './characterType' import { CharacterHair } from './characterHair' import { Item } from './item' +import { UUID } from '#application/types' @Entity() export class Sprite extends BaseEntity { @@ -31,4 +32,40 @@ export class Sprite extends BaseEntity { @OneToMany(() => Item, (item) => item.sprite) items = new Collection(this) + + setId(id: UUID) { + this.id = id + return this + } + + getId() { + return this.id + } + + setName(name: string) { + this.name = name + return this + } + + getName() { + return this.name + } + + setCreatedAt(createdAt: Date) { + this.createdAt = createdAt + return this + } + + getCreatedAt() { + return this.createdAt + } + + setUpdatedAt(updatedAt: Date) { + this.updatedAt = updatedAt + return this + } + + getUpdatedAt() { + return this.updatedAt + } } diff --git a/src/entities/spriteAction.ts b/src/entities/spriteAction.ts index 4ecb019..4c126a3 100644 --- a/src/entities/spriteAction.ts +++ b/src/entities/spriteAction.ts @@ -2,6 +2,7 @@ import { randomUUID } from 'node:crypto' import { Entity, ManyToOne, PrimaryKey, Property } from '@mikro-orm/core' import { BaseEntity } from '#application/bases/baseEntity' import { Sprite } from './sprite' +import { UUID } from '#application/types' @Entity() export class SpriteAction extends BaseEntity { @@ -15,7 +16,7 @@ export class SpriteAction extends BaseEntity { action!: string @Property({ type: 'json', nullable: true }) - sprites?: any + sprites?: string[] @Property() originX = 0 @@ -37,4 +38,103 @@ export class SpriteAction extends BaseEntity { @Property() frameRate = 0 + + setId(id: UUID) { + this.id = id + return this + } + + getId() { + return this.id + } + + setSprite(sprite: Sprite) { + this.sprite = sprite + return this + } + + getSprite() { + return this.sprite + } + + setAction(action: string) { + this.action = action + return this + } + + getAction() { + return this.action + } + + setSprites(sprites: string[]) { + this.sprites = sprites + return this + } + + getSprites() { + return this.sprites + } + + setOriginX(originX: number) { + this.originX = originX + return this + } + + getOriginX() { + return this.originX + } + + setOriginY(originY: number) { + this.originY = originY + return this + } + + getOriginY() { + return this.originY + } + + setIsAnimated(isAnimated: boolean) { + this.isAnimated = isAnimated + return this + } + + getIsAnimated() { + return this.isAnimated + } + + setIsLooping(isLooping: boolean) { + this.isLooping = isLooping + return this + } + + getIsLooping() { + return this.isLooping + } + + setFrameWidth(frameWidth: number) { + this.frameWidth = frameWidth + return this + } + + getFrameWidth() { + return this.frameWidth + } + + setFrameHeight(frameHeight: number) { + this.frameHeight = frameHeight + return this + } + + getFrameHeight() { + return this.frameHeight + } + + setFrameRate(frameRate: number) { + this.frameRate = frameRate + return this + } + + getFrameRate() { + return this.frameRate + } } diff --git a/src/entities/tile.ts b/src/entities/tile.ts index a2a9b45..05921d8 100644 --- a/src/entities/tile.ts +++ b/src/entities/tile.ts @@ -1,6 +1,7 @@ import { randomUUID } from 'node:crypto' import { Entity, PrimaryKey, Property } from '@mikro-orm/core' import { BaseEntity } from '#application/bases/baseEntity' +import { UUID } from '#application/types' @Entity() export class Tile extends BaseEntity { @@ -18,4 +19,49 @@ export class Tile extends BaseEntity { @Property() updatedAt = new Date() + + setId(id: UUID) { + this.id = id + return this + } + + getId() { + return this.id + } + + setName(name: string) { + this.name = name + return this + } + + getName() { + return this.name + } + + setTags(tags: any) { + this.tags = tags + return this + } + + getTags() { + return this.tags + } + + setCreatedAt(createdAt: Date) { + this.createdAt = createdAt + return this + } + + getCreatedAt() { + return this.createdAt + } + + setUpdatedAt(updatedAt: Date) { + this.updatedAt = updatedAt + return this + } + + getUpdatedAt() { + return this.updatedAt + } } diff --git a/src/entities/user.ts b/src/entities/user.ts index 446c13a..bea56ad 100644 --- a/src/entities/user.ts +++ b/src/entities/user.ts @@ -2,6 +2,7 @@ import { Collection, Entity, OneToMany, PrimaryKey, Property } from '@mikro-orm/ import { BaseEntity } from '#application/bases/baseEntity' import { Character } from './character' import { PasswordResetToken } from './passwordResetToken' +import bcrypt from 'bcryptjs' @Entity() export class User extends BaseEntity { @@ -25,4 +26,68 @@ export class User extends BaseEntity { @OneToMany(() => PasswordResetToken, (token) => token.user) passwordResetTokens = new Collection(this) + + setId(id: number) { + this.id = id + return this + } + + getId() { + return this.id + } + + setUsername(username: string) { + this.username = username + return this + } + + getUsername() { + return this.username + } + + setEmail(email: string) { + this.email = email + return this + } + + getEmail() { + return this.email + } + + setPassword(password: string) { + this.password = bcrypt.hashSync(password, 10) + return this + } + + getPassword() { + return this.password + } + + setOnline(online: boolean) { + this.online = online + return this + } + + getOnline() { + return this.online + } + + setCharacters(characters: Collection) { + this.characters = characters + return this + } + + getCharacters() { + return this.characters + } + + setPasswordResetTokens(passwordResetTokens: Collection) { + this.passwordResetTokens = passwordResetTokens + return this + } + + getPasswordResetTokens() { + return this.passwordResetTokens + return this + } } diff --git a/src/entities/world.ts b/src/entities/world.ts index 28dab9a..116a158 100644 --- a/src/entities/world.ts +++ b/src/entities/world.ts @@ -17,4 +17,49 @@ export class World extends BaseEntity { @Property() fogDensity = 0 + + setDate(date: Date) { + this.date = date + return this + } + + getDate() { + return this.date + } + + setIsRainEnabled(isRainEnabled: boolean) { + this.isRainEnabled = isRainEnabled + return this + } + + getIsRainEnabled() { + return this.isRainEnabled + } + + setRainPercentage(rainPercentage: number) { + this.rainPercentage = rainPercentage + return this + } + + getRainPercentage() { + return this.rainPercentage + } + + setIsFogEnabled(isFogEnabled: boolean) { + this.isFogEnabled = isFogEnabled + return this + } + + getIsFogEnabled() { + return this.isFogEnabled + } + + setFogDensity(fogDensity: number) { + this.fogDensity = fogDensity + return this + } + + getFogDensity() { + return this.fogDensity + } } diff --git a/src/entities/zone.ts b/src/entities/zone.ts index 494f0a6..54a445d 100644 --- a/src/entities/zone.ts +++ b/src/entities/zone.ts @@ -50,4 +50,130 @@ export class Zone extends BaseEntity { @Property() updatedAt = new Date() + + setId(id: number) { + this.id = id + return this + } + + getId() { + return this.id + } + + setName(name: string) { + this.name = name + return this + } + + getName() { + return this.name + } + + setWidth(width: number) { + this.width = width + return this + } + + getWidth() { + return this.width + } + + setHeight(height: number) { + this.height = height + return this + } + + getHeight() { + return this.height + } + + setTiles(tiles: any) { + this.tiles = tiles + return this + } + + getTiles() { + return this.tiles + } + + setPvp(pvp: boolean) { + this.pvp = pvp + return this + } + + getPvp() { + return this.pvp + } + + setZoneEffects(zoneEffects: Collection) { + this.zoneEffects = zoneEffects + return this + } + + getZoneEffects() { + return this.zoneEffects + } + + setZoneEventTiles(zoneEventTiles: Collection) { + this.zoneEventTiles = zoneEventTiles + return this + } + + getZoneEventTiles() { + return this.zoneEventTiles + } + + setZoneEventTileTeleports(zoneEventTileTeleports: Collection) { + this.zoneEventTileTeleports = zoneEventTileTeleports + return this + } + + getZoneEventTileTeleports() { + return this.zoneEventTileTeleports + } + + setZoneObjects(zoneObjects: Collection) { + this.zoneObjects = zoneObjects + return this + } + + getZoneObjects() { + return this.zoneObjects + } + + setCharacters(characters: Collection) { + this.characters = characters + return this + } + + getCharacters() { + return this.characters + } + + setChats(chats: Collection) { + this.chats = chats + return this + } + + getChats() { + return this.chats + } + + setCreatedAt(createdAt: Date) { + this.createdAt = createdAt + return this + } + + getCreatedAt() { + return this.createdAt + } + + setUpdatedAt(updatedAt: Date) { + this.updatedAt = updatedAt + return this + } + + getUpdatedAt() { + return this.updatedAt + } } diff --git a/src/entities/zoneEffect.ts b/src/entities/zoneEffect.ts index d34e810..aefbc26 100644 --- a/src/entities/zoneEffect.ts +++ b/src/entities/zoneEffect.ts @@ -2,6 +2,7 @@ import { randomUUID } from 'node:crypto' import { Entity, ManyToOne, PrimaryKey, Property } from '@mikro-orm/core' import { BaseEntity } from '#application/bases/baseEntity' import { Zone } from './zone' +import { UUID } from '#application/types' @Entity() export class ZoneEffect extends BaseEntity { @@ -16,4 +17,40 @@ export class ZoneEffect extends BaseEntity { @Property() strength!: number + + setId(id: UUID) { + this.id = id + return this + } + + getId() { + return this.id + } + + setZone(zone: Zone) { + this.zone = zone + return this + } + + getZone() { + return this.zone + } + + setEffect(effect: string) { + this.effect = effect + return this + } + + getEffect() { + return this.effect + } + + setStrength(strength: number) { + this.strength = strength + return this + } + + getStrength() { + return this.strength + } } diff --git a/src/entities/zoneEventTile.ts b/src/entities/zoneEventTile.ts index 0387e4f..88b768f 100644 --- a/src/entities/zoneEventTile.ts +++ b/src/entities/zoneEventTile.ts @@ -3,11 +3,13 @@ import { BaseEntity } from '#application/bases/baseEntity' import { Zone } from './zone' import { ZoneEventTileType } from '#application/enums' import { ZoneEventTileTeleport } from './zoneEventTileTeleport' +import { UUID } from '#application/types' +import { randomUUID } from 'node:crypto' @Entity() export class ZoneEventTile extends BaseEntity { @PrimaryKey() - id!: string + id = randomUUID() @ManyToOne(() => Zone) zone!: Zone @@ -23,4 +25,58 @@ export class ZoneEventTile extends BaseEntity { @OneToOne(() => ZoneEventTileTeleport, (teleport) => teleport.zoneEventTile) teleport?: ZoneEventTileTeleport + + setId(id: UUID) { + this.id = id + return this + } + + getId() { + return this.id + } + + setZone(zone: Zone) { + this.zone = zone + return this + } + + getZone() { + return this.zone + } + + setType(type: ZoneEventTileType) { + this.type = type + return this + } + + getType() { + return this.type + } + + setPositionX(positionX: number) { + this.positionX = positionX + return this + } + + getPositionX() { + return this.positionX + } + + setPositionY(positionY: number) { + this.positionY = positionY + return this + } + + getPositionY() { + return this.positionY + } + + setTeleport(teleport: ZoneEventTileTeleport) { + this.teleport = teleport + return this + } + + getTeleport() { + return this.teleport + } } diff --git a/src/entities/zoneEventTileTeleport.ts b/src/entities/zoneEventTileTeleport.ts index 3c84c29..cba9fd8 100644 --- a/src/entities/zoneEventTileTeleport.ts +++ b/src/entities/zoneEventTileTeleport.ts @@ -3,6 +3,7 @@ import { Entity, ManyToOne, OneToOne, PrimaryKey, Property } from '@mikro-orm/co import { BaseEntity } from '#application/bases/baseEntity' import { Zone } from './zone' import { ZoneEventTile } from './zoneEventTile' +import { UUID } from '#application/types' @Entity() export class ZoneEventTileTeleport extends BaseEntity { @@ -23,4 +24,58 @@ export class ZoneEventTileTeleport extends BaseEntity { @Property() toPositionY!: number + + setId(id: UUID) { + this.id = id + return this + } + + getId() { + return this.id + } + + setZoneEventTile(zoneEventTile: ZoneEventTile) { + this.zoneEventTile = zoneEventTile + return this + } + + getZoneEventTile() { + return this.zoneEventTile + } + + setToZone(toZone: Zone) { + this.toZone = toZone + return this + } + + getToZone() { + return this.toZone + } + + setToRotation(toRotation: number) { + this.toRotation = toRotation + return this + } + + getToRotation() { + return this.toRotation + } + + setToPositionX(toPositionX: number) { + this.toPositionX = toPositionX + return this + } + + getToPositionX() { + return this.toPositionX + } + + setToPositionY(toPositionY: number) { + this.toPositionY = toPositionY + return this + } + + getToPositionY() { + return this.toPositionY + } } diff --git a/src/entities/zoneObject.ts b/src/entities/zoneObject.ts index 98f5f06..8b3ade0 100644 --- a/src/entities/zoneObject.ts +++ b/src/entities/zoneObject.ts @@ -2,12 +2,14 @@ import { Entity, ManyToOne, PrimaryKey, Property } from '@mikro-orm/core' import { BaseEntity } from '#application/bases/baseEntity' import { Zone } from './zone' import { MapObject } from '#entities/mapObject' +import { UUID } from '#application/types' +import { randomUUID } from 'node:crypto' //@TODO : Rename mapObject @Entity() export class ZoneObject extends BaseEntity { @PrimaryKey() - id!: string + id = randomUUID() @ManyToOne(() => Zone) zone!: Zone @@ -26,4 +28,67 @@ export class ZoneObject extends BaseEntity { @Property() positionY = 0 + + setId(id: UUID) { + this.id = id + return this + } + + getId() { + return this.id + } + + setZone(zone: Zone) { + this.zone = zone + return this + } + + getZone() { + return this.zone + } + + setMapObject(mapObject: MapObject) { + this.mapObject = mapObject + return this + } + + getMapObject() { + return this.mapObject + } + + setDepth(depth: number) { + this.depth = depth + return this + } + + getDepth() { + return this.depth + } + + setIsRotated(isRotated: boolean) { + this.isRotated = isRotated + return this + } + + getIsRotated() { + return this.isRotated + } + + setPositionX(positionX: number) { + this.positionX = positionX + return this + } + + getPositionX() { + return this.positionX + } + + setPositionY(positionY: number) { + this.positionY = positionY + return this + } + + getPositionY() { + return this.positionY + } } diff --git a/src/http/avatar.ts b/src/http/avatar.ts index 7bb40b4..460176f 100644 --- a/src/http/avatar.ts +++ b/src/http/avatar.ts @@ -19,11 +19,11 @@ interface AvatarOptions { async function generateAvatar(res: Response, options: AvatarOptions) { try { const characterType = await CharacterTypeRepository.getById(options.characterTypeId) - if (!characterType?.spriteId) { + if (!characterType?.sprite?.id) { return res.status(404).json({ message: 'Character type not found' }) } - const bodySpritePath = getPublicPath('sprites', characterType.spriteId, 'idle_right_down.png') + const bodySpritePath = getPublicPath('sprites', characterType.sprite.id, 'idle_right_down.png') if (!fs.existsSync(bodySpritePath)) { console.error(`Body sprite file not found: ${bodySpritePath}`) return res.status(404).json({ message: 'Body sprite file not found' }) @@ -37,8 +37,8 @@ async function generateAvatar(res: Response, options: AvatarOptions) { if (options.characterHairId) { const characterHair = await CharacterHairRepository.getById(options.characterHairId) - if (characterHair?.spriteId) { - const hairSpritePath = getPublicPath('sprites', characterHair.spriteId, 'front.png') + if (characterHair?.sprite?.id) { + const hairSpritePath = getPublicPath('sprites', characterHair.sprite.id, 'front.png') if (fs.existsSync(hairSpritePath)) { avatar = avatar.composite([ { diff --git a/src/managers/zoneManager.ts b/src/managers/zoneManager.ts index 8dd7dce..71905a8 100644 --- a/src/managers/zoneManager.ts +++ b/src/managers/zoneManager.ts @@ -1,9 +1,8 @@ -import { Zone } from '@prisma/client' import ZoneRepository from '#repositories/zoneRepository' -import ZoneService from '#services/zoneService' -import LoadedZone from '#models/loadedZone' import { gameLogger } from '#application/logger' import ZoneCharacter from '#models/zoneCharacter' +import LoadedZone from '#models/loadedZone' +import { Zone } from '#entities/zone' class ZoneManager { private readonly zones = new Map() diff --git a/src/models/zoneCharacter.ts b/src/models/zoneCharacter.ts index 3ca0e6f..ecd8978 100644 --- a/src/models/zoneCharacter.ts +++ b/src/models/zoneCharacter.ts @@ -12,7 +12,7 @@ class ZoneCharacter { public async savePosition() { const characterService = new CharacterService() - await characterService.updateCharacterPosition(this.character.id, this.character.positionX, this.character.positionY, this.character.rotation, this.character.zone.id) + await characterService.updateCharacterPosition(this.character.id, this.character.positionX, this.character.positionY, this.character.rotation, this.character.zone!.id) } } diff --git a/src/repositories/worldRepository.ts b/src/repositories/worldRepository.ts index 3e1c4ec..007e64f 100644 --- a/src/repositories/worldRepository.ts +++ b/src/repositories/worldRepository.ts @@ -6,7 +6,7 @@ class WorldRepository extends BaseRepository { async getFirst() { try { const repository = this.em.getRepository(World) - return await repository.findOne({ date: { $exists: true }}) + return await repository.findOne({ date: { $exists: true } }) } catch (error: any) { gameLogger.error(`Failed to get first world: ${error instanceof Error ? error.message : String(error)}`) } diff --git a/src/repositories/zoneRepository.ts b/src/repositories/zoneRepository.ts index 575278b..5c01225 100644 --- a/src/repositories/zoneRepository.ts +++ b/src/repositories/zoneRepository.ts @@ -14,7 +14,7 @@ class ZoneRepository extends BaseRepository { return null } } - + async getAll(): Promise { try { const repository = this.em.getRepository(Zone) diff --git a/src/services/characterService.ts b/src/services/characterService.ts index 234341c..a2baf78 100644 --- a/src/services/characterService.ts +++ b/src/services/characterService.ts @@ -64,7 +64,7 @@ export class CharacterService { character.positionX = positionX character.positionY = positionY character.rotation = rotation - character.zone = await ZoneRepository.getById(zoneId) as Zone + character.zone = (await ZoneRepository.getById(zoneId)) as Zone await character.save() diff --git a/src/services/chatService.ts b/src/services/chatService.ts index 39dd4bf..34dee1d 100644 --- a/src/services/chatService.ts +++ b/src/services/chatService.ts @@ -9,7 +9,6 @@ import CharacterRepository from '#repositories/characterRepository' class ChatService { async sendZoneMessage(io: Server, socket: TSocket, message: string, characterId: number, zoneId: number): Promise { try { - const character = await CharacterRepository.getById(characterId) if (!character) return false @@ -18,9 +17,10 @@ class ChatService { const newChat = new Chat() - newChat.character = character - newChat.zone = zone - newChat.message = message + newChat + .setCharacter(character) + .setZone(zone) + .setMessage(message) await newChat.save() diff --git a/src/services/userService.ts b/src/services/userService.ts index 756bddf..3a7c1f6 100644 --- a/src/services/userService.ts +++ b/src/services/userService.ts @@ -48,9 +48,10 @@ class UserService { const hashedPassword = await bcrypt.hash(password, 10) const newUser = new User() - newUser.username = username - newUser.email = email - newUser.password = hashedPassword + newUser + .setUsername(username) + .setEmail(email) + .setPassword(hashedPassword) await newUser.save() return newUser @@ -81,8 +82,9 @@ class UserService { // Create new token using MikroORM const passwordResetToken = new PasswordResetToken() - passwordResetToken.user = user - passwordResetToken.token = token + passwordResetToken + .setUser(user) + .setToken(token) await passwordResetToken.save() const transporter = NodeMailer.createTransport({ diff --git a/src/services/worldService.ts b/src/services/worldService.ts index 4d45e59..22946d5 100644 --- a/src/services/worldService.ts +++ b/src/services/worldService.ts @@ -10,19 +10,19 @@ class WorldService { world = new World() } - world.date = worldData.date || new Date() + world.setDate(worldData.date || new Date()) if (worldData.isRainEnabled) { - world.isRainEnabled = worldData.isRainEnabled + world.setIsRainEnabled(worldData.isRainEnabled) } if (worldData.rainPercentage) { - world.rainPercentage = worldData.rainPercentage + world.setRainPercentage(worldData.rainPercentage) } if (worldData.isFogEnabled) { - world.isFogEnabled = worldData.isFogEnabled + world.setIsFogEnabled(worldData.isFogEnabled) } if (worldData.fogDensity) { - world.fogDensity = worldData.fogDensity + world.setFogDensity(worldData.fogDensity) } await world.save() diff --git a/src/services/zoneEventTileService.ts b/src/services/zoneEventTileService.ts index 7573d9c..4dcc9e7 100644 --- a/src/services/zoneEventTileService.ts +++ b/src/services/zoneEventTileService.ts @@ -6,7 +6,7 @@ import { ZoneEventTileTeleport } from '#entities/zoneEventTileTeleport' export class ZoneEventTileService { public async handleTeleport(io: Server, socket: TSocket, character: ExtendedCharacter, teleport: ZoneEventTileTeleport): Promise { - if (teleport.toZone.id === character.zone.id) return + if (teleport.toZone.id === character.zone!.id) return const loadedZone = ZoneManager.getZoneById(teleport.toZone.id) if (!loadedZone) { @@ -16,15 +16,16 @@ export class ZoneEventTileService { const zone = loadedZone.getZone() - const oldZoneId = character.zone.id + const oldZoneId = character.zone!.id const newZoneId = teleport.toZone.id - // Update local character object - character.zone = teleport.toZone - character.rotation = teleport.toRotation - character.positionX = teleport.toPositionX - character.positionY = teleport.toPositionY character.isMoving = false + // Update local character object + character + .setZone(teleport.toZone) + .setRotation(teleport.toRotation) + .setPositionX(teleport.toPositionX) + .setPositionY(teleport.toPositionY) await character.save() diff --git a/src/socketEvents/disconnect.ts b/src/socketEvents/disconnect.ts index 425c28a..e3ac326 100644 --- a/src/socketEvents/disconnect.ts +++ b/src/socketEvents/disconnect.ts @@ -38,7 +38,7 @@ export default class DisconnectEvent { gameLogger.info('User disconnected along with their character') // Inform other clients that the character has left - this.io.in(character.zoneId.toString()).emit('zone:character:leave', character.id) + this.io.in(character.zone!.id.toString()).emit('zone:character:leave', character.id) this.io.emit('character:disconnect', character.id) } catch (error: any) { gameLogger.error('disconnect error', error.message)