1
0
forked from noxious/server

Added getter & setter functions to all entities

This commit is contained in:
Dennis Postma 2024-12-26 15:15:18 +01:00
parent 6a27ccff31
commit 691abb7c4f
33 changed files with 1500 additions and 168 deletions

View File

@ -3,6 +3,8 @@ import { Character } from '#entities/character'
import { ZoneEventTile } from '#entities/zoneEventTile' import { ZoneEventTile } from '#entities/zoneEventTile'
import { ZoneEventTileTeleport } from '#entities/zoneEventTileTeleport' import { ZoneEventTileTeleport } from '#entities/zoneEventTileTeleport'
export type UUID = `${string}-${string}-${string}-${string}-${string}`
export type TSocket = Socket & { export type TSocket = Socket & {
userId?: number userId?: number
characterId?: number characterId?: number

View File

@ -2,7 +2,6 @@ import { Server } from 'socket.io'
import fs from 'fs' import fs from 'fs'
import { getPublicPath } from '#application/storage' import { getPublicPath } from '#application/storage'
import sharp from 'sharp' import sharp from 'sharp'
import bcrypt from 'bcryptjs'
import { Tile } from '#entities/tile' import { Tile } from '#entities/tile'
import { MapObject } from '#entities/mapObject' import { MapObject } from '#entities/mapObject'
import { SpriteAction } from '#entities/spriteAction' import { SpriteAction } from '#entities/spriteAction'
@ -17,6 +16,7 @@ import ZoneRepository from '#repositories/zoneRepository'
import { CharacterType } from '#entities/characterType' import { CharacterType } from '#entities/characterType'
import { CharacterGender, CharacterRace } from '#application/enums' import { CharacterGender, CharacterRace } from '#application/enums'
import { CharacterHair } from '#entities/characterHair' import { CharacterHair } from '#entities/characterHair'
import { UUID } from '#application/types'
// @TODO : Replace this with seeding // @TODO : Replace this with seeding
// https://mikro-orm.io/docs/seeding // https://mikro-orm.io/docs/seeding
@ -45,8 +45,8 @@ export default class InitCommand {
private async importTiles(): Promise<void> { private async importTiles(): Promise<void> {
for (const tile of fs.readdirSync(getPublicPath('tiles'))) { for (const tile of fs.readdirSync(getPublicPath('tiles'))) {
const newTile = new Tile() const newTile = new Tile()
newTile.id = tile.split('.')[0] as any newTile.setId(tile.split('.')[0] as UUID).setName('New tile')
newTile.name = 'New tile'
await newTile.save() await newTile.save()
} }
} }
@ -54,16 +54,19 @@ export default class InitCommand {
private async importObjects(): Promise<void> { private async importObjects(): Promise<void> {
for (const object of fs.readdirSync(getPublicPath('objects'))) { for (const object of fs.readdirSync(getPublicPath('objects'))) {
const newMapObject = new MapObject() const newMapObject = new MapObject()
newMapObject.id = object.split('.')[0] as any newMapObject
newMapObject.name = 'New object' .setId(object.split('.')[0] as UUID)
.setName('New object')
newMapObject.frameWidth = await sharp(getPublicPath('objects', object)) .setFrameWidth(
(await sharp(getPublicPath('objects', object))
.metadata() .metadata()
.then((metadata) => metadata.height) ?? 0 .then((metadata) => metadata.height)) ?? 0
)
newMapObject.frameHeight = await sharp(getPublicPath('objects', object)) .setFrameHeight(
(await sharp(getPublicPath('objects', object))
.metadata() .metadata()
.then((metadata) => metadata.width) ?? 0 .then((metadata) => metadata.width)) ?? 0
)
await newMapObject.save() await newMapObject.save()
} }
@ -71,129 +74,123 @@ export default class InitCommand {
private async createCharacterType(): Promise<void> { private async createCharacterType(): Promise<void> {
const characterSprite = new Sprite() const characterSprite = new Sprite()
characterSprite.id = '023d1e9d-f57f-4faa-8412-86c07107cf85' characterSprite.setId('023d1e9d-f57f-4faa-8412-86c07107cf85').setName('Character')
characterSprite.name = 'Character'
await characterSprite.save() await characterSprite.save()
const idleRightDownAction = new SpriteAction() const idleRightDownAction = new SpriteAction()
idleRightDownAction.action = 'idle_right_down' idleRightDownAction
idleRightDownAction.sprites = [ .setAction('idle_right_down')
"" .setSprites([
] // Base64 sprite data ''
idleRightDownAction.originX = 0 ])
idleRightDownAction.originY = 0 .setOriginX(0)
idleRightDownAction.isAnimated = false .setOriginY(0)
idleRightDownAction.isLooping = false .setIsAnimated(false)
idleRightDownAction.frameWidth = 64 .setIsLooping(false)
idleRightDownAction.frameHeight = 94 .setFrameWidth(64)
idleRightDownAction.frameRate = 0 .setFrameHeight(94)
idleRightDownAction.sprite = characterSprite .setFrameRate(0)
.setSprite(characterSprite)
await idleRightDownAction.save() await idleRightDownAction.save()
const idleLeftUpAction = new SpriteAction() const idleLeftUpAction = new SpriteAction()
idleLeftUpAction.action = 'idle_left_up' idleLeftUpAction
idleLeftUpAction.sprites = [ .setAction('idle_left_up')
"" .setSprites([
] // Base64 sprite data ''
idleLeftUpAction.originX = 0 ])
idleLeftUpAction.originY = 0 .setOriginX(0)
idleLeftUpAction.isAnimated = false .setOriginY(0)
idleLeftUpAction.isLooping = false .setIsAnimated(false)
idleLeftUpAction.frameWidth = 64 .setIsLooping(false)
idleLeftUpAction.frameHeight = 94 .setFrameWidth(64)
idleLeftUpAction.frameRate = 0 .setFrameHeight(94)
idleLeftUpAction.sprite = characterSprite .setFrameRate(0)
.setSprite(characterSprite)
await idleLeftUpAction.save() await idleLeftUpAction.save()
const walkRightDownAction = new SpriteAction() const walkRightDownAction = new SpriteAction()
walkRightDownAction
walkRightDownAction.action = 'walk_right_down' .setAction('walk_right_down')
walkRightDownAction.sprites = [ .setSprites([
"", '',
"", '',
"", '',
"" ''
] // Base64 sprite data array ])
walkRightDownAction.originX = 0 .setOriginX(0)
walkRightDownAction.originY = 0 .setOriginY(0)
walkRightDownAction.isAnimated = true .setIsAnimated(true)
walkRightDownAction.isLooping = false .setIsLooping(false)
walkRightDownAction.frameWidth = 64 .setFrameWidth(64)
walkRightDownAction.frameHeight = 94 .setFrameHeight(94)
walkRightDownAction.frameRate = 7 .setFrameRate(7)
walkRightDownAction.sprite = characterSprite .setSprite(characterSprite)
await walkRightDownAction.save() await walkRightDownAction.save()
const walkLeftUpAction = new SpriteAction() const walkLeftUpAction = new SpriteAction()
walkLeftUpAction.action = 'walk_left_up' walkLeftUpAction
walkLeftUpAction.sprites = [ .setAction('walk_left_up')
"", .setSprites([
"", '',
"", '',
"" '',
] // Base64 sprite data array ''
walkLeftUpAction.originX = 0 ])
walkLeftUpAction.originY = 0 .setOriginX(0)
walkLeftUpAction.isAnimated = true .setOriginY(0)
walkLeftUpAction.isLooping = false .setIsAnimated(true)
walkLeftUpAction.frameWidth = 64 .setIsLooping(false)
walkLeftUpAction.frameHeight = 94 .setFrameWidth(64)
walkLeftUpAction.frameRate = 7 .setFrameHeight(94)
walkLeftUpAction.sprite = characterSprite .setFrameRate(7)
.setSprite(characterSprite)
await walkLeftUpAction.save() await walkLeftUpAction.save()
const characterType = new CharacterType() const characterType = new CharacterType()
characterType.id = 1 characterType.setId(1).setName('New character type').setGender(CharacterGender.MALE).setRace(CharacterRace.HUMAN).setIsSelectable(true).setSprite(characterSprite)
characterType.name = 'New character type'
characterType.gender = CharacterGender.MALE
characterType.race = CharacterRace.HUMAN
characterType.isSelectable = true
characterType.sprite = characterSprite
await characterType.save() await characterType.save()
} }
private async createCharacterHair(): Promise<void> { private async createCharacterHair(): Promise<void> {
const hairSprite = new Sprite() const hairSprite = new Sprite()
hairSprite.id = '922ee95f-1500-49c0-8ead-f8cc46dad136' hairSprite.setId('922ee95f-1500-49c0-8ead-f8cc46dad136').setName('Hair 1')
hairSprite.name = 'Hair 1'
await hairSprite.save() await hairSprite.save()
const frontAction = new SpriteAction() const frontAction = new SpriteAction()
frontAction.action = 'front' frontAction
frontAction.sprites = [ .setAction('front')
"" .setSprites([
] // Base64 sprite data ''
frontAction.originX = 0.5 ])
frontAction.originY = 5.34 .setOriginX(0.5)
frontAction.isAnimated = false .setOriginY(5.34)
frontAction.isLooping = false .setIsAnimated(false)
frontAction.frameWidth = 64 .setIsLooping(false)
frontAction.frameHeight = 18 .setFrameWidth(64)
frontAction.frameRate = 0 .setFrameHeight(18)
frontAction.sprite = hairSprite .setFrameRate(0)
.setSprite(hairSprite)
await frontAction.save() await frontAction.save()
const backAction = new SpriteAction() const backAction = new SpriteAction()
backAction.action = 'back' backAction
backAction.sprites = [ .setAction('back')
"" .setSprites([
] // Base64 sprite data ''
backAction.originX = 0.5 ])
backAction.originY = 4.34 .setOriginX(0.5)
backAction.isAnimated = false .setOriginY(4.34)
backAction.isLooping = false .setIsAnimated(false)
backAction.frameWidth = 64 .setIsLooping(false)
backAction.frameHeight = 22 .setFrameWidth(64)
backAction.frameRate = 0 .setFrameHeight(22)
backAction.sprite = hairSprite .setFrameRate(0)
.setSprite(hairSprite)
await backAction.save() await backAction.save()
const characterHair = new CharacterHair() const characterHair = new CharacterHair()
characterHair.id = 1 characterHair.setId(1).setName('Hair 1').setGender(CharacterGender.MALE).setIsSelectable(true).setSprite(hairSprite)
characterHair.name = 'Hair 1'
characterHair.gender = CharacterGender.MALE
characterHair.isSelectable = true
characterHair.sprite = hairSprite
await characterHair.save() await characterHair.save()
} }
@ -215,7 +212,7 @@ export default class InitCommand {
frameWidth: 64, frameWidth: 64,
frameHeight: 94, frameHeight: 94,
frameRate: 0 frameRate: 0
}, }
// Add other actions... // Add other actions...
] ]
@ -231,38 +228,32 @@ export default class InitCommand {
private async createZone(): Promise<void> { private async createZone(): Promise<void> {
const zone = new Zone() const zone = new Zone()
zone.name = 'New zone' zone
zone.width = 100 .setName('New zone')
zone.height = 100 .setWidth(100)
zone.tiles = Array.from({ length: 100 }, () => .setHeight(100)
Array.from({ length: 100 }, () => 'a2fd8d6f-5042-437a-9c1e-c66b91ecc35b') .setTiles(Array.from({ length: 100 }, () => Array.from({ length: 100 }, () => 'a2fd8d6f-5042-437a-9c1e-c66b91ecc35b')))
)
await zone.save() await zone.save()
const effect = new ZoneEffect() const effect = new ZoneEffect()
effect.effect = 'light' effect.setEffect('light').setStrength(100).setZone(zone)
effect.strength = 100
effect.zone = zone
await effect.save() await effect.save()
} }
private async createUser(): Promise<void> { private async createUser(): Promise<void> {
const user = new User() const user = new User()
user.id = 1 user.setId(1).setUsername('root').setEmail('local@host').setPassword('password').setOnline(false)
user.username = 'root'
user.email = 'local@host'
user.password = await bcrypt.hash('password', 10)
user.online = false
await user.save() await user.save()
const character = new Character() const character = new Character()
character.id = 1 character
character.user = user .setId(1)
character.name = 'root' .setUser(user)
character.role = 'gm' .setName('root')
character.zone = await ZoneRepository.getFirst() ?? undefined .setRole('gm')
character.characterType = await CharacterTypeRepository.getFirst() ?? undefined .setZone((await ZoneRepository.getFirst()) ?? undefined)
character.characterHair = await CharacterHairRepository.getFirst() ?? undefined .setCharacterType((await CharacterTypeRepository.getFirst()) ?? undefined)
.setCharacterHair((await CharacterHairRepository.getFirst()) ?? undefined)
await character.save() await character.save()
} }
} }

View File

@ -82,4 +82,211 @@ export class Character extends BaseEntity {
@Property() @Property()
wisdom = 10 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<Chat>) {
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<CharacterItem>) {
this.items = items
return this
}
getItems() {
return this.items
}
setEquipment(equipment: Collection<CharacterEquipment>) {
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
}
} }

View File

@ -17,4 +17,40 @@ export class CharacterEquipment extends BaseEntity {
@ManyToOne(() => CharacterItem) @ManyToOne(() => CharacterItem)
characterItem!: 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
}
} }

View File

@ -23,4 +23,58 @@ export class CharacterHair extends BaseEntity {
@OneToMany(() => Character, (character) => character.characterHair) @OneToMany(() => Character, (character) => character.characterHair)
characters = new Collection<Character>(this) characters = new Collection<Character>(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<Character>) {
this.characters = characters
return this
}
getCharacters() {
return this.characters
}
} }

View File

@ -20,4 +20,49 @@ export class CharacterItem extends BaseEntity {
@OneToMany(() => CharacterEquipment, (equipment) => equipment.characterItem) @OneToMany(() => CharacterEquipment, (equipment) => equipment.characterItem)
characterEquipment = new Collection<CharacterEquipment>(this) characterEquipment = new Collection<CharacterEquipment>(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<CharacterEquipment>) {
this.characterEquipment = characterEquipment
return this
}
getCharacterEquipment() {
return this.characterEquipment
}
} }

View File

@ -32,4 +32,85 @@ export class CharacterType extends BaseEntity {
@Property() @Property()
updatedAt = new Date() 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<Character>) {
this.characters = characters
return this
}
getCharacters() {
return this.characters
}
} }

View File

@ -19,4 +19,49 @@ export class Chat extends BaseEntity {
@Property() @Property()
createdAt = new Date() 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
}
} }

View File

@ -3,11 +3,13 @@ import { BaseEntity } from '#application/bases/baseEntity'
import { Sprite } from './sprite' import { Sprite } from './sprite'
import { CharacterItem } from './characterItem' import { CharacterItem } from './characterItem'
import { ItemType, ItemRarity } from '#application/enums' import { ItemType, ItemRarity } from '#application/enums'
import { randomUUID } from 'node:crypto'
import { UUID } from '#application/types'
@Entity() @Entity()
export class Item extends BaseEntity { export class Item extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id!: string id = randomUUID()
@Property() @Property()
name!: string name!: string
@ -35,4 +37,94 @@ export class Item extends BaseEntity {
@OneToMany(() => CharacterItem, (characterItem) => characterItem.item) @OneToMany(() => CharacterItem, (characterItem) => characterItem.item)
characters = new Collection<CharacterItem>(this) characters = new Collection<CharacterItem>(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<CharacterItem>) {
this.characters = characters
return this
}
getCharacters() {
return this.characters
}
} }

View File

@ -2,6 +2,7 @@ import { randomUUID } from 'node:crypto'
import { Collection, Entity, OneToMany, PrimaryKey, Property } from '@mikro-orm/core' import { Collection, Entity, OneToMany, PrimaryKey, Property } from '@mikro-orm/core'
import { BaseEntity } from '#application/bases/baseEntity' import { BaseEntity } from '#application/bases/baseEntity'
import { ZoneObject } from './zoneObject' import { ZoneObject } from './zoneObject'
import { UUID } from '#application/types'
@Entity() @Entity()
export class MapObject extends BaseEntity { export class MapObject extends BaseEntity {
@ -40,4 +41,112 @@ export class MapObject extends BaseEntity {
@OneToMany(() => ZoneObject, (zoneObject) => zoneObject.mapObject) @OneToMany(() => ZoneObject, (zoneObject) => zoneObject.mapObject)
zoneObjects = new Collection<ZoneObject>(this) zoneObjects = new Collection<ZoneObject>(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<ZoneObject>) {
this.zoneObjects = zoneObjects
return this
}
getZoneObjects() {
return this.zoneObjects
}
} }

View File

@ -15,4 +15,40 @@ export class PasswordResetToken extends BaseEntity {
@Property() @Property()
createdAt = new Date() 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
}
} }

View File

@ -5,6 +5,7 @@ import { SpriteAction } from './spriteAction'
import { CharacterType } from './characterType' import { CharacterType } from './characterType'
import { CharacterHair } from './characterHair' import { CharacterHair } from './characterHair'
import { Item } from './item' import { Item } from './item'
import { UUID } from '#application/types'
@Entity() @Entity()
export class Sprite extends BaseEntity { export class Sprite extends BaseEntity {
@ -31,4 +32,40 @@ export class Sprite extends BaseEntity {
@OneToMany(() => Item, (item) => item.sprite) @OneToMany(() => Item, (item) => item.sprite)
items = new Collection<Item>(this) items = new Collection<Item>(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
}
} }

View File

@ -2,6 +2,7 @@ import { randomUUID } from 'node:crypto'
import { Entity, ManyToOne, PrimaryKey, Property } from '@mikro-orm/core' import { Entity, ManyToOne, PrimaryKey, Property } from '@mikro-orm/core'
import { BaseEntity } from '#application/bases/baseEntity' import { BaseEntity } from '#application/bases/baseEntity'
import { Sprite } from './sprite' import { Sprite } from './sprite'
import { UUID } from '#application/types'
@Entity() @Entity()
export class SpriteAction extends BaseEntity { export class SpriteAction extends BaseEntity {
@ -15,7 +16,7 @@ export class SpriteAction extends BaseEntity {
action!: string action!: string
@Property({ type: 'json', nullable: true }) @Property({ type: 'json', nullable: true })
sprites?: any sprites?: string[]
@Property() @Property()
originX = 0 originX = 0
@ -37,4 +38,103 @@ export class SpriteAction extends BaseEntity {
@Property() @Property()
frameRate = 0 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
}
} }

View File

@ -1,6 +1,7 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { Entity, PrimaryKey, Property } from '@mikro-orm/core' import { Entity, PrimaryKey, Property } from '@mikro-orm/core'
import { BaseEntity } from '#application/bases/baseEntity' import { BaseEntity } from '#application/bases/baseEntity'
import { UUID } from '#application/types'
@Entity() @Entity()
export class Tile extends BaseEntity { export class Tile extends BaseEntity {
@ -18,4 +19,49 @@ export class Tile extends BaseEntity {
@Property() @Property()
updatedAt = new Date() 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
}
} }

View File

@ -2,6 +2,7 @@ import { Collection, Entity, OneToMany, PrimaryKey, Property } from '@mikro-orm/
import { BaseEntity } from '#application/bases/baseEntity' import { BaseEntity } from '#application/bases/baseEntity'
import { Character } from './character' import { Character } from './character'
import { PasswordResetToken } from './passwordResetToken' import { PasswordResetToken } from './passwordResetToken'
import bcrypt from 'bcryptjs'
@Entity() @Entity()
export class User extends BaseEntity { export class User extends BaseEntity {
@ -25,4 +26,68 @@ export class User extends BaseEntity {
@OneToMany(() => PasswordResetToken, (token) => token.user) @OneToMany(() => PasswordResetToken, (token) => token.user)
passwordResetTokens = new Collection<PasswordResetToken>(this) passwordResetTokens = new Collection<PasswordResetToken>(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<Character>) {
this.characters = characters
return this
}
getCharacters() {
return this.characters
}
setPasswordResetTokens(passwordResetTokens: Collection<PasswordResetToken>) {
this.passwordResetTokens = passwordResetTokens
return this
}
getPasswordResetTokens() {
return this.passwordResetTokens
return this
}
} }

View File

@ -17,4 +17,49 @@ export class World extends BaseEntity {
@Property() @Property()
fogDensity = 0 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
}
} }

View File

@ -50,4 +50,130 @@ export class Zone extends BaseEntity {
@Property() @Property()
updatedAt = new Date() 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<ZoneEffect>) {
this.zoneEffects = zoneEffects
return this
}
getZoneEffects() {
return this.zoneEffects
}
setZoneEventTiles(zoneEventTiles: Collection<ZoneEventTile>) {
this.zoneEventTiles = zoneEventTiles
return this
}
getZoneEventTiles() {
return this.zoneEventTiles
}
setZoneEventTileTeleports(zoneEventTileTeleports: Collection<ZoneEventTileTeleport>) {
this.zoneEventTileTeleports = zoneEventTileTeleports
return this
}
getZoneEventTileTeleports() {
return this.zoneEventTileTeleports
}
setZoneObjects(zoneObjects: Collection<ZoneObject>) {
this.zoneObjects = zoneObjects
return this
}
getZoneObjects() {
return this.zoneObjects
}
setCharacters(characters: Collection<Character>) {
this.characters = characters
return this
}
getCharacters() {
return this.characters
}
setChats(chats: Collection<Chat>) {
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
}
} }

View File

@ -2,6 +2,7 @@ import { randomUUID } from 'node:crypto'
import { Entity, ManyToOne, PrimaryKey, Property } from '@mikro-orm/core' import { Entity, ManyToOne, PrimaryKey, Property } from '@mikro-orm/core'
import { BaseEntity } from '#application/bases/baseEntity' import { BaseEntity } from '#application/bases/baseEntity'
import { Zone } from './zone' import { Zone } from './zone'
import { UUID } from '#application/types'
@Entity() @Entity()
export class ZoneEffect extends BaseEntity { export class ZoneEffect extends BaseEntity {
@ -16,4 +17,40 @@ export class ZoneEffect extends BaseEntity {
@Property() @Property()
strength!: number 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
}
} }

View File

@ -3,11 +3,13 @@ import { BaseEntity } from '#application/bases/baseEntity'
import { Zone } from './zone' import { Zone } from './zone'
import { ZoneEventTileType } from '#application/enums' import { ZoneEventTileType } from '#application/enums'
import { ZoneEventTileTeleport } from './zoneEventTileTeleport' import { ZoneEventTileTeleport } from './zoneEventTileTeleport'
import { UUID } from '#application/types'
import { randomUUID } from 'node:crypto'
@Entity() @Entity()
export class ZoneEventTile extends BaseEntity { export class ZoneEventTile extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id!: string id = randomUUID()
@ManyToOne(() => Zone) @ManyToOne(() => Zone)
zone!: Zone zone!: Zone
@ -23,4 +25,58 @@ export class ZoneEventTile extends BaseEntity {
@OneToOne(() => ZoneEventTileTeleport, (teleport) => teleport.zoneEventTile) @OneToOne(() => ZoneEventTileTeleport, (teleport) => teleport.zoneEventTile)
teleport?: ZoneEventTileTeleport 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
}
} }

View File

@ -3,6 +3,7 @@ import { Entity, ManyToOne, OneToOne, PrimaryKey, Property } from '@mikro-orm/co
import { BaseEntity } from '#application/bases/baseEntity' import { BaseEntity } from '#application/bases/baseEntity'
import { Zone } from './zone' import { Zone } from './zone'
import { ZoneEventTile } from './zoneEventTile' import { ZoneEventTile } from './zoneEventTile'
import { UUID } from '#application/types'
@Entity() @Entity()
export class ZoneEventTileTeleport extends BaseEntity { export class ZoneEventTileTeleport extends BaseEntity {
@ -23,4 +24,58 @@ export class ZoneEventTileTeleport extends BaseEntity {
@Property() @Property()
toPositionY!: number 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
}
} }

View File

@ -2,12 +2,14 @@ import { Entity, ManyToOne, PrimaryKey, Property } from '@mikro-orm/core'
import { BaseEntity } from '#application/bases/baseEntity' import { BaseEntity } from '#application/bases/baseEntity'
import { Zone } from './zone' import { Zone } from './zone'
import { MapObject } from '#entities/mapObject' import { MapObject } from '#entities/mapObject'
import { UUID } from '#application/types'
import { randomUUID } from 'node:crypto'
//@TODO : Rename mapObject //@TODO : Rename mapObject
@Entity() @Entity()
export class ZoneObject extends BaseEntity { export class ZoneObject extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id!: string id = randomUUID()
@ManyToOne(() => Zone) @ManyToOne(() => Zone)
zone!: Zone zone!: Zone
@ -26,4 +28,67 @@ export class ZoneObject extends BaseEntity {
@Property() @Property()
positionY = 0 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
}
} }

View File

@ -19,11 +19,11 @@ interface AvatarOptions {
async function generateAvatar(res: Response, options: AvatarOptions) { async function generateAvatar(res: Response, options: AvatarOptions) {
try { try {
const characterType = await CharacterTypeRepository.getById(options.characterTypeId) const characterType = await CharacterTypeRepository.getById(options.characterTypeId)
if (!characterType?.spriteId) { if (!characterType?.sprite?.id) {
return res.status(404).json({ message: 'Character type not found' }) 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)) { if (!fs.existsSync(bodySpritePath)) {
console.error(`Body sprite file not found: ${bodySpritePath}`) console.error(`Body sprite file not found: ${bodySpritePath}`)
return res.status(404).json({ message: 'Body sprite file not found' }) 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) { if (options.characterHairId) {
const characterHair = await CharacterHairRepository.getById(options.characterHairId) const characterHair = await CharacterHairRepository.getById(options.characterHairId)
if (characterHair?.spriteId) { if (characterHair?.sprite?.id) {
const hairSpritePath = getPublicPath('sprites', characterHair.spriteId, 'front.png') const hairSpritePath = getPublicPath('sprites', characterHair.sprite.id, 'front.png')
if (fs.existsSync(hairSpritePath)) { if (fs.existsSync(hairSpritePath)) {
avatar = avatar.composite([ avatar = avatar.composite([
{ {

View File

@ -1,9 +1,8 @@
import { Zone } from '@prisma/client'
import ZoneRepository from '#repositories/zoneRepository' import ZoneRepository from '#repositories/zoneRepository'
import ZoneService from '#services/zoneService'
import LoadedZone from '#models/loadedZone'
import { gameLogger } from '#application/logger' import { gameLogger } from '#application/logger'
import ZoneCharacter from '#models/zoneCharacter' import ZoneCharacter from '#models/zoneCharacter'
import LoadedZone from '#models/loadedZone'
import { Zone } from '#entities/zone'
class ZoneManager { class ZoneManager {
private readonly zones = new Map<number, LoadedZone>() private readonly zones = new Map<number, LoadedZone>()

View File

@ -12,7 +12,7 @@ class ZoneCharacter {
public async savePosition() { public async savePosition() {
const characterService = new CharacterService() 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)
} }
} }

View File

@ -64,7 +64,7 @@ export class CharacterService {
character.positionX = positionX character.positionX = positionX
character.positionY = positionY character.positionY = positionY
character.rotation = rotation character.rotation = rotation
character.zone = await ZoneRepository.getById(zoneId) as Zone character.zone = (await ZoneRepository.getById(zoneId)) as Zone
await character.save() await character.save()

View File

@ -9,7 +9,6 @@ import CharacterRepository from '#repositories/characterRepository'
class ChatService { class ChatService {
async sendZoneMessage(io: Server, socket: TSocket, message: string, characterId: number, zoneId: number): Promise<boolean> { async sendZoneMessage(io: Server, socket: TSocket, message: string, characterId: number, zoneId: number): Promise<boolean> {
try { try {
const character = await CharacterRepository.getById(characterId) const character = await CharacterRepository.getById(characterId)
if (!character) return false if (!character) return false
@ -18,9 +17,10 @@ class ChatService {
const newChat = new Chat() const newChat = new Chat()
newChat.character = character newChat
newChat.zone = zone .setCharacter(character)
newChat.message = message .setZone(zone)
.setMessage(message)
await newChat.save() await newChat.save()

View File

@ -48,9 +48,10 @@ class UserService {
const hashedPassword = await bcrypt.hash(password, 10) const hashedPassword = await bcrypt.hash(password, 10)
const newUser = new User() const newUser = new User()
newUser.username = username newUser
newUser.email = email .setUsername(username)
newUser.password = hashedPassword .setEmail(email)
.setPassword(hashedPassword)
await newUser.save() await newUser.save()
return newUser return newUser
@ -81,8 +82,9 @@ class UserService {
// Create new token using MikroORM // Create new token using MikroORM
const passwordResetToken = new PasswordResetToken() const passwordResetToken = new PasswordResetToken()
passwordResetToken.user = user passwordResetToken
passwordResetToken.token = token .setUser(user)
.setToken(token)
await passwordResetToken.save() await passwordResetToken.save()
const transporter = NodeMailer.createTransport({ const transporter = NodeMailer.createTransport({

View File

@ -10,19 +10,19 @@ class WorldService {
world = new World() world = new World()
} }
world.date = worldData.date || new Date() world.setDate(worldData.date || new Date())
if (worldData.isRainEnabled) { if (worldData.isRainEnabled) {
world.isRainEnabled = worldData.isRainEnabled world.setIsRainEnabled(worldData.isRainEnabled)
} }
if (worldData.rainPercentage) { if (worldData.rainPercentage) {
world.rainPercentage = worldData.rainPercentage world.setRainPercentage(worldData.rainPercentage)
} }
if (worldData.isFogEnabled) { if (worldData.isFogEnabled) {
world.isFogEnabled = worldData.isFogEnabled world.setIsFogEnabled(worldData.isFogEnabled)
} }
if (worldData.fogDensity) { if (worldData.fogDensity) {
world.fogDensity = worldData.fogDensity world.setFogDensity(worldData.fogDensity)
} }
await world.save() await world.save()

View File

@ -6,7 +6,7 @@ import { ZoneEventTileTeleport } from '#entities/zoneEventTileTeleport'
export class ZoneEventTileService { export class ZoneEventTileService {
public async handleTeleport(io: Server, socket: TSocket, character: ExtendedCharacter, teleport: ZoneEventTileTeleport): Promise<void> { public async handleTeleport(io: Server, socket: TSocket, character: ExtendedCharacter, teleport: ZoneEventTileTeleport): Promise<void> {
if (teleport.toZone.id === character.zone.id) return if (teleport.toZone.id === character.zone!.id) return
const loadedZone = ZoneManager.getZoneById(teleport.toZone.id) const loadedZone = ZoneManager.getZoneById(teleport.toZone.id)
if (!loadedZone) { if (!loadedZone) {
@ -16,15 +16,16 @@ export class ZoneEventTileService {
const zone = loadedZone.getZone() const zone = loadedZone.getZone()
const oldZoneId = character.zone.id const oldZoneId = character.zone!.id
const newZoneId = teleport.toZone.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 character.isMoving = false
// Update local character object
character
.setZone(teleport.toZone)
.setRotation(teleport.toRotation)
.setPositionX(teleport.toPositionX)
.setPositionY(teleport.toPositionY)
await character.save() await character.save()

View File

@ -38,7 +38,7 @@ export default class DisconnectEvent {
gameLogger.info('User disconnected along with their character') gameLogger.info('User disconnected along with their character')
// Inform other clients that the character has left // 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) this.io.emit('character:disconnect', character.id)
} catch (error: any) { } catch (error: any) {
gameLogger.error('disconnect error', error.message) gameLogger.error('disconnect error', error.message)