Compare commits

..

No commits in common. "main" and "v0.0.6" have entirely different histories.
main ... v0.0.6

155 changed files with 4899 additions and 3036 deletions

17
.dockerignore Normal file
View File

@ -0,0 +1,17 @@
node_modules
npm-debug.log
Dockerfile
.dockerignore
.git
.gitignore
README.md
.env
.env.*
docker-compose*
*.md
coverage
.vscode
.idea
dist
build
temp

View File

@ -6,10 +6,10 @@ JWT_SECRET="secret"
CLIENT_URL="http://localhost:5173" CLIENT_URL="http://localhost:5173"
# Database configuration # Database configuration
REDIS_URL="redis://@127.0.0.1:6379/4" REDIS_URL="redis://@redis:6379/4"
DB_HOST="localhost" DB_HOST="mariadb"
DB_USER="root" DB_USER="mariadb"
DB_PASS="" DB_PASS="mariadb"
DB_PORT="3306" DB_PORT="3306"
DB_NAME="game" DB_NAME="game"
@ -26,7 +26,3 @@ SMTP_HOST=my.directonline.io
SMTP_PORT=587 SMTP_PORT=587
SMTP_USER=no-reply@noxious.gg SMTP_USER=no-reply@noxious.gg
SMTP_PASSWORD="" SMTP_PASSWORD=""
# SSL
#PUBLIC_KEY_PATH=
#PRIVATE_KEY_PATH=

View File

@ -1 +0,0 @@
migrations

View File

@ -4,8 +4,5 @@
"tabWidth": 2, "tabWidth": 2,
"singleQuote": true, "singleQuote": true,
"printWidth": 200, "printWidth": 200,
"trailingComma": "none", "trailingComma": "none"
"plugins": ["@ianvs/prettier-plugin-sort-imports"],
"importOrderParserPlugins": ["typescript", "jsx", "decorators-legacy", "classProperties"],
"importOrderCaseSensitive": false
} }

16
Dockerfile Normal file
View File

@ -0,0 +1,16 @@
FROM node:lts-alpine
# Install packages
RUN apk update
RUN apk add --no-cache tmux coreutils
WORKDIR /usr/src/app
COPY package*.json ./
RUN npm ci
COPY . .
# Modify CMD to use tmux
CMD npx mikro-orm-esm migration:up && npm run start

View File

@ -5,7 +5,7 @@ This is the server for the Noxious game.
## Projects requirements ## Projects requirements
- NodeJS 20.x or higher - NodeJS 20.x or higher
- MariaDB 11.x or higher - MySQL 8.x or higher
- Redis 7.x or higher - Redis 7.x or higher
## Installation ## Installation
@ -14,7 +14,7 @@ This is the server for the Noxious game.
2. Install dependencies with `npm install` 2. Install dependencies with `npm install`
3. Copy the `.env.example` file to `.env` and fill in the required variables 3. Copy the `.env.example` file to `.env` and fill in the required variables
4. Extract assets.zip to the `public` folder 4. Extract assets.zip to the `public` folder
5. After MySQL and Redis are running, run `npm run mikro-orm migration:up` to create the database schema 5. After MySQL and Redis are running, run `npx mikro-orm-esm migration:up` to create the database schema
6. Run the server with `npm run dev` 6. Run the server with `npm run dev`
7. Write `init` in the console to import default data and restart the server 7. Write `init` in the console to import default data and restart the server
8. Write `tiles` in the console to fix tile sizes 8. Write `tiles` in the console to fix tile sizes
@ -39,15 +39,15 @@ MikroORM is used as the ORM for the server.
### Create init. migrations ### Create init. migrations
Run `npm run mikro-orm migration:create --initial` to create a new initial migration. Run `npx mikro-orm-esm migration:create --initial` to create a new initial migration.
### Create migrations ### Create migrations
Run `npm run mikro-orm migration:create` to create a new migration. You do this when you want to add a new table or change an existing one. Run `npx mikro-orm-esm migration:create` to create a new migration. You do this when you want to add a new table or change an existing one.
### Apply migrations ### Apply migrations
Run `npm run mikro-orm migration:up` to apply all pending migrations. Run `npx mikro-orm-esm migration:up` to apply all pending migrations.
### Import default data ### Import default data

95
docker-compose.yml Normal file
View File

@ -0,0 +1,95 @@
services:
app:
build:
context: .
dockerfile: Dockerfile
ports:
- "${PORT}:${PORT}"
environment:
- ENV=${ENV}
- HOST=${HOST}
- PORT=${PORT}
- JWT_SECRET=${JWT_SECRET}
- CLIENT_URL=${CLIENT_URL}
- REDIS_URL=${REDIS_URL}
- DB_HOST=${DB_HOST}
- DB_USER=${DB_USER}
- DB_PASS=${DB_PASS}
- DB_PORT=${DB_PORT}
- DB_NAME=${DB_NAME}
- ALLOW_DIAGONAL_MOVEMENT=${ALLOW_DIAGONAL_MOVEMENT}
- DEFAULT_CHARACTER_ZONE=${DEFAULT_CHARACTER_ZONE}
- DEFAULT_CHARACTER_POS_X=${DEFAULT_CHARACTER_POS_X}
- DEFAULT_CHARACTER_POS_Y=${DEFAULT_CHARACTER_POS_Y}
- SMTP_HOST=${SMTP_HOST}
- SMTP_PORT=${SMTP_PORT}
- SMTP_USER=${SMTP_USER}
- SMTP_PASSWORD=${SMTP_PASSWORD}
volumes:
- app-public:/user/src/app/public
- app-logs:/user/src/app/logs
depends_on:
- mariadb
- redis
restart: unless-stopped
networks:
- app-network
labels:
- "traefik.enable=true"
- "traefik.http.routers.app.rule=Host(`${HOST}`)"
- "traefik.http.routers.app.entrypoints=websecure"
- "traefik.http.routers.app.tls.certresolver=le"
- "traefik.http.services.app.loadbalancer.server.port=${PORT}"
- "traefik.http.routers.app.middlewares=websocket"
traefik:
image: traefik:v3.3.3
ports:
- "80:80"
- "443:443"
- "8080:8080"
volumes:
- traefik_data:/data
- ./traefik.yml:/etc/traefik/traefik.yml
- /var/run/docker.sock:/var/run/docker.sock:ro
restart: unless-stopped
networks:
- app-network
mariadb:
image: mariadb:lts
environment:
- MARIADB_USER=${DB_USER}
- MARIADB_PASSWORD=${DB_PASS}
- MARIADB_DATABASE=${DB_NAME}
- MARIADB_RANDOM_ROOT_PASSWORD=yes
volumes:
- mariadb-data:/var/lib/mysql
ports:
- "${DB_PORT}:3306"
restart: unless-stopped
networks:
- app-network
command: [ 'mariadbd', '--character-set-server=utf8mb4', '--collation-server=utf8mb4_unicode_ci' ]
redis:
image: redis:7.4.2-alpine
command: redis-server --appendonly yes
volumes:
- redis-data:/data
ports:
- "6379:6379"
restart: unless-stopped
networks:
- app-network
networks:
app-network:
driver: bridge
volumes:
app-public:
app-logs:
mariadb-data:
redis-data:
traefik_data:

42
eslint.config.js Normal file
View File

@ -0,0 +1,42 @@
import eslint from '@eslint/js';
import tseslint from '@typescript-eslint/eslint-plugin';
import tsparser from '@typescript-eslint/parser';
import importPlugin from 'eslint-plugin-import';
export default [
eslint.configs.recommended,
{
files: ['**/*.ts'],
languageOptions: {
parser: tsparser,
parserOptions: {
project: './tsconfig.json',
ecmaVersion: 2023,
},
},
plugins: {
'@typescript-eslint': tseslint,
'import': importPlugin,
},
rules: {
...tseslint.configs['recommended'].rules,
...tseslint.configs['recommended-requiring-type-checking'].rules,
'import/order': ['error', {
'groups': [
'builtin',
'external',
'internal',
['parent', 'sibling'],
'index',
'object',
'type'
],
'newlines-between': 'always',
'alphabetize': {
'order': 'asc',
'caseInsensitive': true
}
}]
}
}
];

3605
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@ -5,23 +5,36 @@
"start": "tsx src/server.ts", "start": "tsx src/server.ts",
"dev": "nodemon --exec tsx src/server.ts", "dev": "nodemon --exec tsx src/server.ts",
"format": "prettier --write src/", "format": "prettier --write src/",
"mikro-orm": "tsx ./node_modules/.bin/mikro-orm-esm" "lint": "eslint .",
"lint:fix": "eslint . --fix"
},
"imports": {
"#root/*": "./src/*.js",
"#application/*": "./src/application/*.js",
"#commands/*": "./src/commands/*.js",
"#entities/*": "./src/entities/*.js",
"#controllers/*": "./src/controllers/*.js",
"#jobs/*": "./src/jobs/*.js",
"#managers/*": "./src/managers/*.js",
"#middleware/*": "./src/middleware/*.js",
"#models/*": "./src/models/*.js",
"#repositories/*": "./src/repositories/*.js",
"#services/*": "./src/services/*.js",
"#events/*": "./src/events/*.js"
}, },
"dependencies": { "dependencies": {
"@mikro-orm/cli": "^6.4.2",
"@mikro-orm/core": "^6.4.2", "@mikro-orm/core": "^6.4.2",
"@mikro-orm/mariadb": "^6.4.2", "@mikro-orm/mariadb": "^6.4.2",
"@mikro-orm/migrations": "^6.4.2", "@mikro-orm/migrations": "^6.4.2",
"@mikro-orm/mysql": "^6.4.2", "@mikro-orm/mysql": "^6.4.2",
"@mikro-orm/reflection": "^6.4.2", "@mikro-orm/reflection": "^6.4.2",
"@mikro-orm/cli": "^6.4.2",
"@types/ioredis": "^4.28.10", "@types/ioredis": "^4.28.10",
"bcryptjs": "^2.4.3", "bcryptjs": "^2.4.3",
"bufferutil": "^4.0.9",
"bullmq": "^5.13.2", "bullmq": "^5.13.2",
"cors": "^2.8.5", "cors": "^2.8.5",
"dotenv": "^16.4.5", "dotenv": "^16.4.5",
"express": "^4.19.2", "express": "^4.19.2",
"https": "^1.0.0",
"ioredis": "^5.4.1", "ioredis": "^5.4.1",
"jsonwebtoken": "^9.0.2", "jsonwebtoken": "^9.0.2",
"nodemailer": "^6.9.15", "nodemailer": "^6.9.15",
@ -29,18 +42,21 @@
"reflect-metadata": "^0.2.2", "reflect-metadata": "^0.2.2",
"sharp": "^0.33.4", "sharp": "^0.33.4",
"socket.io": "^4.7.5", "socket.io": "^4.7.5",
"ts-node": "^10.9.2",
"typescript": "^5.5.3", "typescript": "^5.5.3",
"utf-8-validate": "^6.0.5",
"zod": "^3.23.8" "zod": "^3.23.8"
}, },
"devDependencies": { "devDependencies": {
"ts-node": "^10.9.2",
"@ianvs/prettier-plugin-sort-imports": "^4.4.0",
"@types/bcryptjs": "^2.4.6", "@types/bcryptjs": "^2.4.6",
"@types/express": "^4.17.21", "@types/express": "^4.17.21",
"@types/jsonwebtoken": "^9.0.6", "@types/jsonwebtoken": "^9.0.6",
"@types/node": "^20.14.11", "@types/node": "^20.14.11",
"@types/nodemailer": "^6.4.16", "@types/nodemailer": "^6.4.16",
"@typescript-eslint/eslint-plugin": "^8.18.2",
"@typescript-eslint/parser": "^8.18.2",
"eslint": "^9.17.0",
"eslint-config-prettier": "^9.1.0",
"eslint-plugin-import": "^2.31.0",
"nodemon": "^3.1.4", "nodemon": "^3.1.4",
"prettier": "^3.3.3", "prettier": "^3.3.3",
"tsx": "^4.19.2" "tsx": "^4.19.2"

Binary file not shown.

View File

@ -1,6 +1,7 @@
import Logger, { LoggerType } from '@/application/logger'
import { Server } from 'socket.io' import { Server } from 'socket.io'
import Logger, { LoggerType } from '#application/logger'
export abstract class BaseCommand { export abstract class BaseCommand {
protected readonly logger = Logger.type(LoggerType.COMMAND) protected readonly logger = Logger.type(LoggerType.COMMAND)
constructor(readonly io: Server) {} constructor(readonly io: Server) {}

View File

@ -1,7 +1,9 @@
import fs from 'fs' import fs from 'fs'
import Logger, { LoggerType } from '@/application/logger'
import type { Response } from 'express' import type { Response } from 'express'
import Logger, { LoggerType } from '#application/logger'
export abstract class BaseController { export abstract class BaseController {
protected readonly logger = Logger.type(LoggerType.HTTP) protected readonly logger = Logger.type(LoggerType.HTTP)

View File

@ -1,7 +1,8 @@
import Database from '@/application/database'
import Logger, { LoggerType } from '@/application/logger'
import { EntityManager } from '@mikro-orm/core' import { EntityManager } from '@mikro-orm/core'
import Database from '#application/database'
import Logger, { LoggerType } from '#application/logger'
export abstract class BaseEntity { export abstract class BaseEntity {
protected readonly logger = Logger.type(LoggerType.ENTITY) protected readonly logger = Logger.type(LoggerType.ENTITY)
protected entityManager?: EntityManager protected entityManager?: EntityManager

View File

@ -1,43 +1,22 @@
import { SocketEvent } from '@/application/enums'
import Logger, { LoggerType } from '@/application/logger'
import type { TSocket } from '@/application/types'
import { Character } from '@/entities/character'
import MapManager from '@/managers/mapManager'
import type MapCharacter from '@/models/mapCharacter'
import CharacterRepository from '@/repositories/characterRepository'
import { Server } from 'socket.io' import { Server } from 'socket.io'
import type { TSocket } from '#application/types'
import Logger, { LoggerType } from '#application/logger'
import { Character } from '#entities/character'
import CharacterRepository from '#repositories/characterRepository'
export abstract class BaseEvent { export abstract class BaseEvent {
protected readonly logger = Logger.type(LoggerType.GAME) protected readonly logger = Logger.type(LoggerType.GAME)
private lastActionTimes: Map<string, number> = new Map()
constructor( constructor(
readonly io: Server, readonly io: Server,
readonly socket: TSocket readonly socket: TSocket
) {} ) {}
protected isThrottled(actionId: string, throttleTime: number): boolean {
const now = Date.now()
const lastActionTime = this.lastActionTimes.get(actionId) || 0
if (now - lastActionTime < throttleTime) {
return true
}
this.lastActionTimes.set(actionId, now)
return false
}
protected getMapCharacter(): MapCharacter | null {
if (!this.socket.characterId) return null
return MapManager.getCharacterById(this.socket.characterId)
}
protected async getCharacter(): Promise<Character | null> { protected async getCharacter(): Promise<Character | null> {
if (!this.socket.characterId) return null
const characterRepository = new CharacterRepository() const characterRepository = new CharacterRepository()
return characterRepository.getById(this.socket.characterId) return characterRepository.getById(this.socket.characterId!)
} }
protected async isCharacterGM(): Promise<boolean> { protected async isCharacterGM(): Promise<boolean> {
@ -45,16 +24,15 @@ export abstract class BaseEvent {
return character?.getRole() === 'gm' return character?.getRole() === 'gm'
} }
protected sendNotificationAndLog(message: string): void { protected emitError(message: string): void {
console.log(message) this.socket.emit('notification', { title: 'Server message', message })
this.socket.emit(SocketEvent.NOTIFICATION, { title: 'Server message', message }) this.logger.error('Base event error', `Player ${this.socket.userId}: ${message}`)
this.logger.error('Base event error' + `Player ${this.socket.userId}: ${message}`)
} }
protected handleError(context: string, error: unknown): void { protected handleError(context: string, error: unknown): void {
console.log(error) console.log(error)
const errorMessage = error instanceof Error ? error.message : error && typeof error === 'object' && 'toString' in error ? error.toString() : String(error) const errorMessage = error instanceof Error ? error.message : error && typeof error === 'object' && 'toString' in error ? error.toString() : String(error)
this.socket.emit(SocketEvent.NOTIFICATION, { title: 'Server message', message: `Server error occured. Please contact the server administrator.` }) this.socket.emit('notification', { title: 'Server message', message: `Server error occured. Please contact the server administrator.` })
this.logger.error('Base event error: ' + errorMessage) this.logger.error('Base event error', errorMessage)
} }
} }

View File

@ -1,7 +1,8 @@
import Database from '@/application/database'
import Logger, { LoggerType } from '@/application/logger'
import { EntityManager } from '@mikro-orm/core' import { EntityManager } from '@mikro-orm/core'
import Database from '#application/database'
import Logger, { LoggerType } from '#application/logger'
export abstract class BaseRepository { export abstract class BaseRepository {
protected readonly logger = Logger.type(LoggerType.REPOSITORY) protected readonly logger = Logger.type(LoggerType.REPOSITORY)
private entityManager?: EntityManager private entityManager?: EntityManager

View File

@ -1,4 +1,4 @@
import Logger, { LoggerType } from '@/application/logger' import Logger, { LoggerType } from '#application/logger'
export abstract class BaseService { export abstract class BaseService {
protected readonly logger = Logger.type(LoggerType.GAME) protected readonly logger = Logger.type(LoggerType.GAME)

View File

@ -6,7 +6,7 @@ class config {
// Server configuration // Server configuration
static ENV: string = process.env.ENV || 'development' static ENV: string = process.env.ENV || 'development'
static HOST: string = process.env.HOST || '0.0.0.0' static HOST: string = process.env.HOST || '0.0.0.0'
static PORT: number = process.env.PORT ? parseInt(process.env.PORT) : 4000 static PORT: number = process.env.PORT ? parseInt(process.env.PORT) : 6969
static JWT_SECRET: string = process.env.JWT_SECRET || 'secret' static JWT_SECRET: string = process.env.JWT_SECRET || 'secret'
static CLIENT_URL: string = process.env.CLIENT_URL ? process.env.CLIENT_URL : 'https://noxious.gg' static CLIENT_URL: string = process.env.CLIENT_URL ? process.env.CLIENT_URL : 'https://noxious.gg'
@ -31,10 +31,6 @@ class config {
static SMTP_PORT: number = process.env.SMTP_PORT ? parseInt(process.env.SMTP_PORT) : 587 static SMTP_PORT: number = process.env.SMTP_PORT ? parseInt(process.env.SMTP_PORT) : 587
static SMTP_USER: string = process.env.SMTP_USER || 'no-reply@noxious.gg' static SMTP_USER: string = process.env.SMTP_USER || 'no-reply@noxious.gg'
static SMTP_PASSWORD: string = process.env.SMTP_PASSWORD || 'password' static SMTP_PASSWORD: string = process.env.SMTP_PASSWORD || 'password'
// SSL
static PUBLIC_KEY_PATH: string = process.env.PUBLIC_KEY_PATH || ''
static PRIVATE_KEY_PATH: string = process.env.PRIVATE_KEY_PATH || ''
} }
export default config export default config

View File

@ -1,9 +1,11 @@
import * as fs from 'fs' import * as fs from 'fs'
import * as path from 'path' import * as path from 'path'
import { pathToFileURL } from 'url' import { pathToFileURL } from 'url'
import Logger, { LoggerType } from '@/application/logger'
import Storage from '@/application/storage' import type { Command } from '#application/types'
import type { Command } from '@/application/types'
import Logger, { LoggerType } from '#application/logger'
import Storage from '#application/storage'
export class CommandRegistry { export class CommandRegistry {
private readonly commands: Map<string, Command> = new Map() private readonly commands: Map<string, Command> = new Map()

View File

@ -1,6 +1,7 @@
import * as fs from 'fs' import * as fs from 'fs'
import * as path from 'path' import * as path from 'path'
import Logger, { LoggerType } from '@/application/logger'
import Logger, { LoggerType } from '#application/logger'
export class LogReader { export class LogReader {
private logger = Logger.type(LoggerType.CONSOLE) private logger = Logger.type(LoggerType.CONSOLE)

View File

@ -1,8 +1,7 @@
// import { MikroORM } from '@mikro-orm/mysql' import { MikroORM } from '@mikro-orm/mysql'
import Logger, { LoggerType } from '@/application/logger' import Logger, { LoggerType } from '#application/logger'
import config from '@/root/mikro-orm.config' import config from '#root/mikro-orm.config'
import { MikroORM } from '@mikro-orm/mariadb'
class Database { class Database {
private static orm: MikroORM private static orm: MikroORM

View File

@ -1,59 +1,17 @@
export enum SocketEvent { export enum SocketEvent {
CONNECT_ERROR = 'connect_error', CHARACTER_CONNECT = 1,
RECONNECT_FAILED = 'reconnect_failed', CHARACTER_MOVE = 2,
CLOSE = '52', CHARACTER_MOVE_ERROR = 3,
DATA = '51', CHARACTER_TELEPORT = 4,
CHARACTER_CONNECT = '50', ZONE_CHARACTER_LEAVE = 5,
CHARACTER_CREATE = '49', ZONE_CHARACTER_JOIN = 6,
CHARACTER_DELETE = '48', ZONE_CHARACTER_LIST = 7,
CHARACTER_LIST = '47', ZONE_CHARACTER_DELETE = 8,
GM_CHARACTERHAIR_CREATE = '46', ZONE_CHARACTER_CREATE = 9,
GM_CHARACTERHAIR_REMOVE = '45', ZONE_CHARACTER_UPDATE = 10,
GM_CHARACTERHAIR_LIST = '44', ZONE_CHARACTER_HAIR_UPDATE = 11,
GM_CHARACTERHAIR_UPDATE = '43', ZONE_CHARACTER_HAIR_LIST = 12,
GM_CHARACTERTYPE_CREATE = '42', ZONE_CHARACTER_TELEPORT = 13
GM_CHARACTERTYPE_REMOVE = '41',
GM_CHARACTERTYPE_LIST = '40',
GM_CHARACTERTYPE_UPDATE = '39',
GM_ITEM_CREATE = '38',
GM_ITEM_REMOVE = '37',
GM_ITEM_LIST = '36',
GM_ITEM_UPDATE = '35',
GM_MAPOBJECT_LIST = '34',
GM_MAPOBJECT_REMOVE = '33',
GM_MAPOBJECT_UPDATE = '32',
GM_MAPOBJECT_UPLOAD = '31',
GM_SPRITE_COPY = '30',
GM_SPRITE_CREATE = '29',
GM_SPRITE_DELETE = '28',
GM_SPRITE_LIST = '27',
GM_SPRITE_UPDATE = '26',
GM_TILE_DELETE = '25',
GM_TILE_LIST = '24',
GM_TILE_UPDATE = '23',
GM_TILE_UPLOAD = '22',
GM_MAP_CREATE = '21',
GM_MAP_DELETE = '20',
GM_MAP_REQUEST = '19',
GM_MAP_UPDATE = '18',
MAP_CHARACTER_MOVEERROR = '17',
DISCONNECT = 'disconnect',
USER_DISCONNECT = '15',
LOGIN = '14',
LOGGED_IN = '13',
NOTIFICATION = '12',
DATE = '11',
FAILED = '10',
COMPLETED = '9',
CONNECTION = 'connection',
WEATHER = '7',
CHARACTER_DISCONNECT = '6',
MAP_CHARACTER_ATTACK = '5',
MAP_CHARACTER_TELEPORT = '4',
MAP_CHARACTER_JOIN = '3',
MAP_CHARACTER_LEAVE = '2',
MAP_CHARACTER_MOVE = '1',
CHAT_MESSAGE = '0'
} }
export enum ItemType { export enum ItemType {

View File

@ -1,5 +1,4 @@
import pino from 'pino' import pino from 'pino'
const logger = pino.pino const logger = pino.pino
export enum LoggerType { export enum LoggerType {

View File

@ -1,6 +1,7 @@
import fs from 'fs' import fs from 'fs'
import path from 'path' import path from 'path'
import config from '@/application/config'
import config from '#application/config'
class Storage { class Storage {
private readonly baseDir: string private readonly baseDir: string
@ -8,7 +9,7 @@ class Storage {
constructor() { constructor() {
this.rootDir = process.cwd() this.rootDir = process.cwd()
this.baseDir = config.ENV === 'development' ? 'src' : 'src' this.baseDir = config.ENV === 'development' ? 'src' : 'dist'
} }
/** /**

View File

@ -1,8 +1,9 @@
import { Character } from '@/entities/character'
import { MapEventTile } from '@/entities/mapEventTile'
import { MapEventTileTeleport } from '@/entities/mapEventTileTeleport'
import { Server, Socket } from 'socket.io' import { Server, Socket } from 'socket.io'
import { Character } from '#entities/character'
import { MapEventTile } from '#entities/mapEventTile'
import { MapEventTileTeleport } from '#entities/mapEventTileTeleport'
export type UUID = `${string}-${string}-${string}-${string}-${string}` export type UUID = `${string}-${string}-${string}-${string}-${string}`
export type TSocket = Socket & { export type TSocket = Socket & {

View File

@ -58,16 +58,3 @@ export const ZCharacterCreate = z.object({
.max(255, { message: 'Name must be at most 255 characters long' }) .max(255, { message: 'Name must be at most 255 characters long' })
.regex(/^[A-Za-z][A-Za-z0-9_-]*$/, { message: 'Name must start with a letter and can only contain letters, numbers, underscores, or dashes' }) .regex(/^[A-Za-z][A-Za-z0-9_-]*$/, { message: 'Name must start with a letter and can only contain letters, numbers, underscores, or dashes' })
}) })
export const ZCharacterConnect = z.object({
characterId: z.string(),
characterHairId: z.string().optional().nullable(),
newNickname: z
.string()
.min(3, { message: 'Name must be at least 3 characters long' })
.max(255, { message: 'Name must be at most 255 characters long' })
.regex(/^[A-Za-z][A-Za-z0-9_-]*$/, {
message: 'Name must start with a letter and can only contain letters, numbers, underscores, or dashes'
})
.optional()
})

View File

@ -1,13 +1,13 @@
import { BaseCommand } from '@/application/base/baseCommand'
import { SocketEvent } from '@/application/enums'
import { Server } from 'socket.io' import { Server } from 'socket.io'
import { BaseCommand } from '#application/base/baseCommand'
type CommandInput = string[] type CommandInput = string[]
export default class AlertCommand extends BaseCommand { export default class AlertCommand extends BaseCommand {
public execute(input: CommandInput): void { public execute(input: CommandInput): void {
const message: string = input.join(' ') ?? null const message: string = input.join(' ') ?? null
if (!message) return console.log('message is required') if (!message) return console.log('message is required')
this.io.emit(SocketEvent.NOTIFICATION, { message: message }) this.io.emit('notification', { message: message })
} }
} }

View File

@ -1,23 +1,26 @@
import fs from 'fs' import fs from 'fs'
import { BaseCommand } from '@/application/base/baseCommand'
import { CharacterGender, CharacterRace } from '@/application/enums'
import Storage from '@/application/storage'
import type { UUID } from '@/application/types'
import { Character } from '@/entities/character'
import { CharacterHair } from '@/entities/characterHair'
import { CharacterType } from '@/entities/characterType'
import { Map } from '@/entities/map'
import { MapEffect } from '@/entities/mapEffect'
import { MapObject } from '@/entities/mapObject'
import { Sprite } from '@/entities/sprite'
import { SpriteAction } from '@/entities/spriteAction'
import { Tile } from '@/entities/tile'
import { User } from '@/entities/user'
import CharacterHairRepository from '@/repositories/characterHairRepository'
import CharacterTypeRepository from '@/repositories/characterTypeRepository'
import MapRepository from '@/repositories/mapRepository'
import sharp from 'sharp' import sharp from 'sharp'
import type { UUID } from '#application/types'
import { BaseCommand } from '#application/base/baseCommand'
import { CharacterGender, CharacterRace } from '#application/enums'
import Storage from '#application/storage'
import { Character } from '#entities/character'
import { CharacterHair } from '#entities/characterHair'
import { CharacterType } from '#entities/characterType'
import { Map } from '#entities/map'
import { MapEffect } from '#entities/mapEffect'
import { MapObject } from '#entities/mapObject'
import { Sprite } from '#entities/sprite'
import { SpriteAction } from '#entities/spriteAction'
import { Tile } from '#entities/tile'
import { User } from '#entities/user'
import CharacterHairRepository from '#repositories/characterHairRepository'
import CharacterTypeRepository from '#repositories/characterTypeRepository'
import MapRepository from '#repositories/mapRepository'
// @TODO : Replace this with seeding // @TODO : Replace this with seeding
// https://mikro-orm.io/docs/seeding // https://mikro-orm.io/docs/seeding
@ -30,8 +33,7 @@ export default class InitCommand extends BaseCommand {
// Assets // Assets
await this.importTiles() await this.importTiles()
await this.importMapObjects() await this.importMapObjects()
await this.createMaleCharacterType() await this.createCharacterType()
// await this.createFemaleCharacterType()
await this.createCharacterHair() await this.createCharacterHair()
// await this.createCharacterEquipment() // await this.createCharacterEquipment()
@ -75,9 +77,9 @@ export default class InitCommand extends BaseCommand {
} }
} }
private async createMaleCharacterType(): Promise<void> { private async createCharacterType(): Promise<void> {
const characterSprite = new Sprite() const characterSprite = new Sprite()
characterSprite.setId('023d1e9d-f57f-4faa-8412-86c07107cf85').setName('Male character') characterSprite.setId('023d1e9d-f57f-4faa-8412-86c07107cf85').setName('Character')
await characterSprite.save() await characterSprite.save()
const idleRightDownAction = new SpriteAction() const idleRightDownAction = new SpriteAction()
@ -273,213 +275,7 @@ export default class InitCommand extends BaseCommand {
const characterType = new CharacterType() const characterType = new CharacterType()
await characterType await characterType
.setId('75b70c78-17f0-44c0-a4fa-15043cb95be0') .setId('75b70c78-17f0-44c0-a4fa-15043cb95be0')
.setName('Male character') .setName('New character type')
.setGender(CharacterGender.MALE)
.setRace(CharacterRace.HUMAN)
.setIsSelectable(true)
.setSprite(characterSprite)
.save()
}
private async createFemaleCharacterType(): Promise<void> {
const characterSprite = new Sprite()
characterSprite.setId('023d1e9d-f57f-4faa-8412-86c07107cf85').setName('Male character')
await characterSprite.save()
const idleRightDownAction = new SpriteAction()
await idleRightDownAction
.setAction('idle_right_down')
.setSprites([
{
url: '',
offset: {
x: 0,
y: 0
}
}
])
.setOriginX(0)
.setOriginY(0)
.setFrameWidth(28)
.setFrameHeight(94)
.setFrameRate(0)
.setSprite(characterSprite)
.save()
const idleLeftUpAction = new SpriteAction()
await idleLeftUpAction
.setAction('idle_left_up')
.setSprites([
{
url: '',
offset: {
x: 0,
y: 0
}
}
])
.setOriginX(0)
.setOriginY(0)
.setFrameWidth(26)
.setFrameHeight(93)
.setFrameRate(0)
.setSprite(characterSprite)
.save()
const walkRightDownAction = new SpriteAction()
await walkRightDownAction
.setAction('walk_right_down')
.setSprites([
{
url: '',
offset: {
x: 7,
y: 8
}
},
{
url: '',
offset: {
x: 7,
y: 2
}
},
{
url: '',
offset: {
x: 2,
y: 2
}
},
{
url: '',
offset: {
x: 0,
y: 0
}
}
])
.setOriginX(0)
.setOriginY(0)
.setFrameWidth(36)
.setFrameHeight(102)
.setFrameRate(7)
.setSprite(characterSprite)
.save()
const walkLeftUpAction = new SpriteAction()
await walkLeftUpAction
.setAction('walk_left_up')
.setSprites([
{
url: '',
offset: {
x: 3,
y: 2
}
},
{
url: '',
offset: {
x: 0,
y: 2
}
},
{
url: '',
offset: {
x: 5,
y: 6
}
},
{
url: '',
offset: {
x: 2,
y: 6
}
}
])
.setOriginX(0)
.setOriginY(0)
.setFrameWidth(34)
.setFrameHeight(101)
.setFrameRate(7)
.setSprite(characterSprite)
.save()
const attackRightDownAction = new SpriteAction()
await attackRightDownAction
.setAction('attack_right_down')
.setSprites([
{
url: '',
offset: {
x: 20,
y: 0
}
},
{
url: '',
offset: {
x: 19,
y: 8
}
},
{
url: '',
offset: {
x: 17,
y: 3
}
}
])
.setOriginX(0)
.setOriginY(0)
.setFrameWidth(69)
.setFrameHeight(111)
.setFrameRate(5)
.setSprite(characterSprite)
.save()
const attackLeftUpAction = new SpriteAction()
await attackLeftUpAction
.setAction('attack_left_up')
.setSprites([
{
url: '',
offset: {
x: 2,
y: 0
}
},
{
url: '',
offset: {
x: 5,
y: 0
}
},
{
url: '',
offset: {
x: 6,
y: 1
}
}
])
.setOriginX(0)
.setOriginY(0)
.setFrameWidth(34)
.setFrameHeight(100)
.setFrameRate(5)
.setSprite(characterSprite)
.save()
const characterType = new CharacterType()
await characterType
.setId('75b70c78-17f0-44c0-a4fa-15043cb95be0')
.setName('Male character')
.setGender(CharacterGender.MALE) .setGender(CharacterGender.MALE)
.setRace(CharacterRace.HUMAN) .setRace(CharacterRace.HUMAN)
.setIsSelectable(true) .setIsSelectable(true)
@ -489,7 +285,7 @@ export default class InitCommand extends BaseCommand {
private async createCharacterHair(): Promise<void> { private async createCharacterHair(): Promise<void> {
const hairSprite = new Sprite() const hairSprite = new Sprite()
hairSprite.setId('922ee95f-1500-49c0-8ead-f8cc46dad136').setName('Hair 1').setWidth(30).setHeight(40) hairSprite.setId('922ee95f-1500-49c0-8ead-f8cc46dad136').setName('Hair 1')
await hairSprite.save() await hairSprite.save()
const frontAction = new SpriteAction() const frontAction = new SpriteAction()
@ -533,7 +329,7 @@ export default class InitCommand extends BaseCommand {
.save() .save()
const characterHair = new CharacterHair() const characterHair = new CharacterHair()
await characterHair.setId('a2471230-d238-4ffb-9eca-9eab869f1b67').setName('Hair 1').setGender(CharacterGender.MALE).setColor('#1B1212').setIsSelectable(true).setSprite(hairSprite).save() await characterHair.setId('a2471230-d238-4ffb-9eca-9eab869f1b67').setName('Hair 1').setGender(CharacterGender.MALE).setIsSelectable(true).setSprite(hairSprite).save()
} }
private async createMap(): Promise<void> { private async createMap(): Promise<void> {
@ -552,8 +348,6 @@ export default class InitCommand extends BaseCommand {
private async createUser(): Promise<void> { private async createUser(): Promise<void> {
const user = new User() const user = new User()
await user.setId('6f9a58b4-172d-425e-b9ea-71e1d13d81ee').setUsername('root').setEmail('local@host').setPassword('password').setOnline(false).save() await user.setId('6f9a58b4-172d-425e-b9ea-71e1d13d81ee').setUsername('root').setEmail('local@host').setPassword('password').setOnline(false).save()
const map = await this.mapRepository.getFirst()
if (!map) return
const character = new Character() const character = new Character()
await character await character
@ -561,7 +355,7 @@ export default class InitCommand extends BaseCommand {
.setUser(user) .setUser(user)
.setName('root') .setName('root')
.setRole('gm') .setRole('gm')
.setMap(map) .setMap((await this.mapRepository.getFirst())!)
.setCharacterType(await this.characterTypeRepository.getFirst()) .setCharacterType(await this.characterTypeRepository.getFirst())
.setCharacterHair(await this.characterHairRepository.getFirst()) .setCharacterHair(await this.characterHairRepository.getFirst())
.save() .save()

View File

@ -1,5 +1,5 @@
import { BaseCommand } from '@/application/base/baseCommand' import { BaseCommand } from '#application/base/baseCommand'
import MapManager from '@/managers/mapManager' import MapManager from '#managers/mapManager'
type CommandInput = string[] type CommandInput = string[]

View File

@ -1,8 +1,10 @@
import fs from 'fs' import fs from 'fs'
import { BaseCommand } from '@/application/base/baseCommand'
import Storage from '@/application/storage'
import sharp from 'sharp' import sharp from 'sharp'
import { BaseCommand } from '#application/base/baseCommand'
import Storage from '#application/storage'
export default class TilesCommand extends BaseCommand { export default class TilesCommand extends BaseCommand {
public async execute(): Promise<void> { public async execute(): Promise<void> {
// Get all tiles // Get all tiles

View File

@ -1,10 +1,12 @@
import { BaseController } from '@/application/base/baseController'
import config from '@/application/config'
import { loginAccountSchema, newPasswordSchema, registerAccountSchema, resetPasswordSchema } from '@/application/zodTypes'
import UserService from '@/services/userService'
import type { Request, Response } from 'express'
import jwt from 'jsonwebtoken' import jwt from 'jsonwebtoken'
import type { Request, Response } from 'express'
import { BaseController } from '#application/base/baseController'
import config from '#application/config'
import { loginAccountSchema, registerAccountSchema, resetPasswordSchema, newPasswordSchema } from '#application/zodTypes'
import UserService from '#services/userService'
export class AuthController extends BaseController { export class AuthController extends BaseController {
/** /**
* Login user * Login user

View File

@ -1,13 +1,16 @@
import fs from 'fs' import fs from 'fs'
import { BaseController } from '@/application/base/baseController'
import Storage from '@/application/storage'
import type { UUID } from '@/application/types'
import CharacterHairRepository from '@/repositories/characterHairRepository'
import CharacterRepository from '@/repositories/characterRepository'
import CharacterTypeRepository from '@/repositories/characterTypeRepository'
import type { Request, Response } from 'express'
import sharp from 'sharp' import sharp from 'sharp'
import type { UUID } from '#application/types'
import type { Request, Response } from 'express'
import { BaseController } from '#application/base/baseController'
import Storage from '#application/storage'
import CharacterHairRepository from '#repositories/characterHairRepository'
import CharacterRepository from '#repositories/characterRepository'
import CharacterTypeRepository from '#repositories/characterTypeRepository'
interface AvatarOptions { interface AvatarOptions {
characterTypeId: UUID characterTypeId: UUID
characterHairId?: UUID characterHairId?: UUID
@ -65,12 +68,9 @@ export class AvatarController extends BaseController {
return this.sendError(res, 'Body sprite file not found', 404) return this.sendError(res, 'Body sprite file not found', 404)
} }
// Get body sprite metadata
const bodyMetadata = await sharp(bodySpritePath).metadata()
let avatar = sharp(bodySpritePath).extend({ let avatar = sharp(bodySpritePath).extend({
top: 0, top: 2,
bottom: 0, bottom: 2,
background: { r: 0, g: 0, b: 0, alpha: 0 } background: { r: 0, g: 0, b: 0, alpha: 0 }
}) })
@ -79,21 +79,7 @@ export class AvatarController extends BaseController {
if (characterHair?.sprite?.id) { if (characterHair?.sprite?.id) {
const hairSpritePath = Storage.getPublicPath('sprites', characterHair.sprite.id, 'front.png') const hairSpritePath = Storage.getPublicPath('sprites', characterHair.sprite.id, 'front.png')
if (fs.existsSync(hairSpritePath)) { if (fs.existsSync(hairSpritePath)) {
// Resize hair sprite to match body dimensions avatar = avatar.composite([{ input: hairSpritePath, gravity: 'north' }])
const resizedHair = await sharp(hairSpritePath)
.resize(bodyMetadata.width, bodyMetadata.height, {
fit: 'contain',
background: { r: 0, g: 0, b: 0, alpha: 0 }
})
.toBuffer()
avatar = avatar.composite([
{
input: resizedHair,
left: 0,
top: -27 // Apply vertical offset
}
])
} }
} }
} }
@ -101,7 +87,6 @@ export class AvatarController extends BaseController {
res.setHeader('Content-Type', 'image/png') res.setHeader('Content-Type', 'image/png')
return avatar.pipe(res) return avatar.pipe(res)
} catch (error) { } catch (error) {
console.error('Avatar generation error:', error)
return this.sendError(res, 'Error generating avatar', 500) return this.sendError(res, 'Error generating avatar', 500)
} }
} }

View File

@ -1,12 +1,13 @@
import { BaseController } from '@/application/base/baseController'
import CharacterHairRepository from '@/repositories/characterHairRepository'
import CharacterTypeRepository from '@/repositories/characterTypeRepository'
import MapObjectRepository from '@/repositories/mapObjectRepository'
import MapRepository from '@/repositories/mapRepository'
import SpriteRepository from '@/repositories/spriteRepository'
import TileRepository from '@/repositories/tileRepository'
import type { Request, Response } from 'express' import type { Request, Response } from 'express'
import { BaseController } from '#application/base/baseController'
import CharacterHairRepository from '#repositories/characterHairRepository'
import CharacterTypeRepository from '#repositories/characterTypeRepository'
import MapObjectRepository from '#repositories/mapObjectRepository'
import MapRepository from '#repositories/mapRepository'
import SpriteRepository from '#repositories/spriteRepository'
import TileRepository from '#repositories/tileRepository'
export class CacheController extends BaseController { export class CacheController extends BaseController {
/** /**
* Serve a list of tiles and send as JSON * Serve a list of tiles and send as JSON

View File

@ -1,7 +1,8 @@
import { BaseController } from '@/application/base/baseController'
import Storage from '@/application/storage'
import type { Request, Response } from 'express' import type { Request, Response } from 'express'
import { BaseController } from '#application/base/baseController'
import Storage from '#application/storage'
export class TexturesController extends BaseController { export class TexturesController extends BaseController {
/** /**
* Download texture * Download texture

View File

@ -1,15 +1,18 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { BaseEntity } from '@/application/base/baseEntity'
import type { UUID } from '@/application/types'
import type { CharacterEquipment } from '@/entities/characterEquipment'
import type { CharacterHair } from '@/entities/characterHair'
import type { CharacterItem } from '@/entities/characterItem'
import type { CharacterType } from '@/entities/characterType'
import type { Chat } from '@/entities/chat'
import type { Map } from '@/entities/map'
import type { User } from '@/entities/user'
import { Collection, ManyToOne, OneToMany, PrimaryKey, Property } from '@mikro-orm/core' import { Collection, ManyToOne, OneToMany, PrimaryKey, Property } from '@mikro-orm/core'
import type { UUID } from '#application/types'
import type { CharacterEquipment } from '#entities/characterEquipment'
import type { CharacterHair } from '#entities/characterHair'
import type { CharacterItem } from '#entities/characterItem'
import type { CharacterType } from '#entities/characterType'
import type { Chat } from '#entities/chat'
import type { Map } from '#entities/map'
import type { User } from '#entities/user'
import { BaseEntity } from '#application/base/baseEntity'
export class BaseCharacter extends BaseEntity { export class BaseCharacter extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id = randomUUID() id = randomUUID()

View File

@ -1,11 +1,14 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { BaseEntity } from '@/application/base/baseEntity'
import { CharacterEquipmentSlotType } from '@/application/enums'
import type { UUID } from '@/application/types'
import type { Character } from '@/entities/character'
import type { CharacterItem } from '@/entities/characterItem'
import { Enum, ManyToOne, PrimaryKey } from '@mikro-orm/core' import { Enum, ManyToOne, PrimaryKey } from '@mikro-orm/core'
import type { UUID } from '#application/types'
import type { Character } from '#entities/character'
import type { CharacterItem } from '#entities/characterItem'
import { BaseEntity } from '#application/base/baseEntity'
import { CharacterEquipmentSlotType } from '#application/enums'
export class BaseCharacterEquipment extends BaseEntity { export class BaseCharacterEquipment extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id = randomUUID() id = randomUUID()

View File

@ -1,11 +1,14 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { BaseEntity } from '@/application/base/baseEntity'
import { CharacterGender } from '@/application/enums'
import type { UUID } from '@/application/types'
import { Character } from '@/entities/character'
import { Sprite } from '@/entities/sprite'
import { Collection, Entity, ManyToOne, OneToMany, PrimaryKey, Property } from '@mikro-orm/core' import { Collection, Entity, ManyToOne, OneToMany, PrimaryKey, Property } from '@mikro-orm/core'
import type { UUID } from '#application/types'
import { BaseEntity } from '#application/base/baseEntity'
import { CharacterGender } from '#application/enums'
import { Character } from '#entities/character'
import { Sprite } from '#entities/sprite'
export class BaseCharacterHair extends BaseEntity { export class BaseCharacterHair extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id = randomUUID() id = randomUUID()
@ -16,14 +19,11 @@ export class BaseCharacterHair extends BaseEntity {
@Property() @Property()
gender: CharacterGender = CharacterGender.MALE gender: CharacterGender = CharacterGender.MALE
@Property()
color: string = '#000000'
@Property() @Property()
isSelectable = false isSelectable = false
@ManyToOne() @ManyToOne()
sprite!: Sprite sprite?: Sprite
@Property() @Property()
createdAt = new Date() createdAt = new Date()
@ -58,15 +58,6 @@ export class BaseCharacterHair extends BaseEntity {
return this.gender return this.gender
} }
setColor(color: string) {
this.color = color
return this
}
getColor() {
return this.color
}
setIsSelectable(isSelectable: boolean) { setIsSelectable(isSelectable: boolean) {
this.isSelectable = isSelectable this.isSelectable = isSelectable
return this return this

View File

@ -1,10 +1,13 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { BaseEntity } from '@/application/base/baseEntity'
import type { UUID } from '@/application/types'
import type { Character } from '@/entities/character'
import type { Item } from '@/entities/item'
import { ManyToOne, PrimaryKey, Property } from '@mikro-orm/core' import { ManyToOne, PrimaryKey, Property } from '@mikro-orm/core'
import type { UUID } from '#application/types'
import type { Character } from '#entities/character'
import type { Item } from '#entities/item'
import { BaseEntity } from '#application/base/baseEntity'
export class BaseCharacterItem extends BaseEntity { export class BaseCharacterItem extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id = randomUUID() id = randomUUID()

View File

@ -1,11 +1,14 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { BaseEntity } from '@/application/base/baseEntity'
import { CharacterGender, CharacterRace } from '@/application/enums'
import type { UUID } from '@/application/types'
import { Character } from '@/entities/character'
import { Sprite } from '@/entities/sprite'
import { Collection, Entity, Enum, ManyToOne, OneToMany, PrimaryKey, Property } from '@mikro-orm/core' import { Collection, Entity, Enum, ManyToOne, OneToMany, PrimaryKey, Property } from '@mikro-orm/core'
import type { UUID } from '#application/types'
import { BaseEntity } from '#application/base/baseEntity'
import { CharacterGender, CharacterRace } from '#application/enums'
import { Character } from '#entities/character'
import { Sprite } from '#entities/sprite'
export class BaseCharacterType extends BaseEntity { export class BaseCharacterType extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id = randomUUID() id = randomUUID()

View File

@ -1,10 +1,13 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { BaseEntity } from '@/application/base/baseEntity'
import type { UUID } from '@/application/types'
import type { Character } from '@/entities/character'
import type { Map } from '@/entities/map'
import { ManyToOne, PrimaryKey, Property } from '@mikro-orm/core' import { ManyToOne, PrimaryKey, Property } from '@mikro-orm/core'
import type { UUID } from '#application/types'
import type { Character } from '#entities/character'
import type { Map } from '#entities/map'
import { BaseEntity } from '#application/base/baseEntity'
export class BaseChat extends BaseEntity { export class BaseChat extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id = randomUUID() id = randomUUID()

View File

@ -1,11 +1,14 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { BaseEntity } from '@/application/base/baseEntity'
import { ItemRarity, ItemType } from '@/application/enums'
import type { UUID } from '@/application/types'
import { CharacterItem } from '@/entities/characterItem'
import { Sprite } from '@/entities/sprite'
import { Collection, Entity, Enum, ManyToOne, OneToMany, PrimaryKey, Property } from '@mikro-orm/core' import { Collection, Entity, Enum, ManyToOne, OneToMany, PrimaryKey, Property } from '@mikro-orm/core'
import type { UUID } from '#application/types'
import { BaseEntity } from '#application/base/baseEntity'
import { ItemType, ItemRarity } from '#application/enums'
import { CharacterItem } from '#entities/characterItem'
import { Sprite } from '#entities/sprite'
export class BaseItem extends BaseEntity { export class BaseItem extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id = randomUUID() id = randomUUID()
@ -25,8 +28,8 @@ export class BaseItem extends BaseEntity {
@Enum(() => ItemRarity) @Enum(() => ItemRarity)
rarity: ItemRarity = ItemRarity.COMMON rarity: ItemRarity = ItemRarity.COMMON
@ManyToOne() @ManyToOne(() => Sprite)
sprite!: Sprite sprite?: Sprite
@Property() @Property()
createdAt = new Date() createdAt = new Date()

View File

@ -1,17 +1,20 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { BaseEntity } from '@/application/base/baseEntity'
import type { UUID } from '@/application/types'
import type { MapEffect } from '@/entities/mapEffect'
import type { MapEventTile } from '@/entities/mapEventTile'
import type { PlacedMapObject } from '@/entities/placedMapObject'
import { Collection, OneToMany, PrimaryKey, Property } from '@mikro-orm/core' import { Collection, OneToMany, PrimaryKey, Property } from '@mikro-orm/core'
import type { UUID } from '#application/types'
import type { MapEffect } from '#entities/mapEffect'
import type { MapEventTile } from '#entities/mapEventTile'
import type { PlacedMapObject } from '#entities/placedMapObject'
import { BaseEntity } from '#application/base/baseEntity'
export class BaseMap extends BaseEntity { export class BaseMap extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id = randomUUID() id = randomUUID()
@Property() @Property()
name: string = '' name!: string
@Property() @Property()
width = 10 width = 10
@ -19,7 +22,7 @@ export class BaseMap extends BaseEntity {
@Property() @Property()
height = 10 height = 10
@Property({ type: 'json' }) @Property({ type: 'json', nullable: true })
tiles: Array<Array<string>> = [] tiles: Array<Array<string>> = []
@Property() @Property()

View File

@ -1,9 +1,12 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { BaseEntity } from '@/application/base/baseEntity'
import type { UUID } from '@/application/types'
import type { Map } from '@/entities/map'
import { ManyToOne, PrimaryKey, Property } from '@mikro-orm/core' import { ManyToOne, PrimaryKey, Property } from '@mikro-orm/core'
import type { UUID } from '#application/types'
import type { Map } from '#entities/map'
import { BaseEntity } from '#application/base/baseEntity'
export class BaseMapEffect extends BaseEntity { export class BaseMapEffect extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id = randomUUID() id = randomUUID()

View File

@ -1,11 +1,14 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { BaseEntity } from '@/application/base/baseEntity'
import { MapEventTileType } from '@/application/enums'
import type { UUID } from '@/application/types'
import type { Map } from '@/entities/map'
import type { MapEventTileTeleport } from '@/entities/mapEventTileTeleport'
import { Enum, ManyToOne, OneToOne, PrimaryKey, Property } from '@mikro-orm/core' import { Enum, ManyToOne, OneToOne, PrimaryKey, Property } from '@mikro-orm/core'
import type { UUID } from '#application/types'
import type { Map } from '#entities/map'
import type { MapEventTileTeleport } from '#entities/mapEventTileTeleport'
import { BaseEntity } from '#application/base/baseEntity'
import { MapEventTileType } from '#application/enums'
export class BaseMapEventTile extends BaseEntity { export class BaseMapEventTile extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id = randomUUID() id = randomUUID()
@ -17,12 +20,12 @@ export class BaseMapEventTile extends BaseEntity {
type!: MapEventTileType type!: MapEventTileType
@Property() @Property()
positionX: number = 0 positionX!: number
@Property() @Property()
positionY: number = 0 positionY!: number
@OneToOne({ eager: true, deleteRule: 'cascade', orphanRemoval: true }) @OneToOne({ eager: true })
teleport?: MapEventTileTeleport teleport?: MapEventTileTeleport
setId(id: UUID) { setId(id: UUID) {

View File

@ -1,15 +1,18 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { BaseEntity } from '@/application/base/baseEntity'
import type { UUID } from '@/application/types'
import type { Map } from '@/entities/map'
import type { MapEventTile } from '@/entities/mapEventTile'
import { ManyToOne, OneToOne, PrimaryKey, Property } from '@mikro-orm/core' import { ManyToOne, OneToOne, PrimaryKey, Property } from '@mikro-orm/core'
import type { UUID } from '#application/types'
import type { Map } from '#entities/map'
import type { MapEventTile } from '#entities/mapEventTile'
import { BaseEntity } from '#application/base/baseEntity'
export class BaseMapEventTileTeleport extends BaseEntity { export class BaseMapEventTileTeleport extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id = randomUUID() id = randomUUID()
@OneToOne({ deleteRule: 'cascade', orphanRemoval: true }) @OneToOne({ deleteRule: 'cascade' })
mapEventTile!: MapEventTile mapEventTile!: MapEventTile
@ManyToOne({ deleteRule: 'cascade', eager: true }) @ManyToOne({ deleteRule: 'cascade', eager: true })

View File

@ -1,8 +1,11 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { BaseEntity } from '@/application/base/baseEntity'
import type { UUID } from '@/application/types'
import { Entity, PrimaryKey, Property } from '@mikro-orm/core' import { Entity, PrimaryKey, Property } from '@mikro-orm/core'
import type { UUID } from '#application/types'
import { BaseEntity } from '#application/base/baseEntity'
export class BaseMapObject extends BaseEntity { export class BaseMapObject extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id = randomUUID() id = randomUUID()
@ -10,11 +13,8 @@ export class BaseMapObject extends BaseEntity {
@Property() @Property()
name!: string name!: string
@Property({ type: 'json' }) @Property({ type: 'json', nullable: true })
tags: string[] = [] tags?: any
@Property({ type: 'json' })
depthOffsets: number[] = [0]
@Property({ type: 'decimal', precision: 10, scale: 2 }) @Property({ type: 'decimal', precision: 10, scale: 2 })
originX = 0 originX = 0
@ -64,14 +64,6 @@ export class BaseMapObject extends BaseEntity {
return this.tags return this.tags
} }
setDepthOffsets(offsets: number[]) {
this.depthOffsets = offsets
}
getDepthOffsets() {
return this.depthOffsets
}
setOriginX(originX: number) { setOriginX(originX: number) {
this.originX = originX this.originX = originX
return this return this

View File

@ -1,9 +1,12 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { BaseEntity } from '@/application/base/baseEntity'
import type { UUID } from '@/application/types'
import type { User } from '@/entities/user'
import { ManyToOne, PrimaryKey, Property } from '@mikro-orm/core' import { ManyToOne, PrimaryKey, Property } from '@mikro-orm/core'
import type { UUID } from '#application/types'
import type { User } from '#entities/user'
import { BaseEntity } from '#application/base/baseEntity'
export class BasePasswordResetToken extends BaseEntity { export class BasePasswordResetToken extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id = randomUUID() id = randomUUID()

View File

@ -1,10 +1,13 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { BaseEntity } from '@/application/base/baseEntity'
import type { UUID } from '@/application/types'
import type { Map } from '@/entities/map'
import type { MapObject } from '@/entities/mapObject'
import { ManyToOne, PrimaryKey, Property } from '@mikro-orm/core' import { ManyToOne, PrimaryKey, Property } from '@mikro-orm/core'
import type { UUID } from '#application/types'
import type { Map } from '#entities/map'
import type { MapObject } from '#entities/mapObject'
import { BaseEntity } from '#application/base/baseEntity'
export class BasePlacedMapObject extends BaseEntity { export class BasePlacedMapObject extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id = randomUUID() id = randomUUID()

View File

@ -1,9 +1,12 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { BaseEntity } from '@/application/base/baseEntity'
import type { UUID } from '@/application/types'
import { SpriteAction } from '@/entities/spriteAction'
import { Collection, Entity, OneToMany, PrimaryKey, Property } from '@mikro-orm/core' import { Collection, Entity, OneToMany, PrimaryKey, Property } from '@mikro-orm/core'
import type { UUID } from '#application/types'
import { BaseEntity } from '#application/base/baseEntity'
import { SpriteAction } from '#entities/spriteAction'
export class BaseSprite extends BaseEntity { export class BaseSprite extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id = randomUUID() id = randomUUID()
@ -11,15 +14,9 @@ export class BaseSprite extends BaseEntity {
@Property() @Property()
name!: string name!: string
@OneToMany({ mappedBy: 'sprite', orphanRemoval: true }) @OneToMany(() => SpriteAction, (action) => action.sprite)
spriteActions = new Collection<SpriteAction>(this) spriteActions = new Collection<SpriteAction>(this)
@Property({ nullable: true })
width: number | null = null
@Property({ nullable: true })
height: number | null = null
@Property() @Property()
createdAt = new Date() createdAt = new Date()
@ -53,24 +50,6 @@ export class BaseSprite extends BaseEntity {
return this.spriteActions return this.spriteActions
} }
setWidth(width: number | null) {
this.width = width
return this
}
getWidth() {
return this.width
}
setHeight(height: number | null) {
this.height = height
return this
}
getHeight() {
return this.height
}
setCreatedAt(createdAt: Date) { setCreatedAt(createdAt: Date) {
this.createdAt = createdAt this.createdAt = createdAt
return this return this

View File

@ -1,9 +1,12 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { BaseEntity } from '@/application/base/baseEntity'
import type { UUID } from '@/application/types'
import type { Sprite } from '@/entities/sprite'
import { ManyToOne, PrimaryKey, Property } from '@mikro-orm/core' import { ManyToOne, PrimaryKey, Property } from '@mikro-orm/core'
import type { UUID } from '#application/types'
import type { Sprite } from '#entities/sprite'
import { BaseEntity } from '#application/base/baseEntity'
export interface SpriteImage { export interface SpriteImage {
url: string url: string
offset: { offset: {

View File

@ -1,8 +1,11 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { BaseEntity } from '@/application/base/baseEntity'
import type { UUID } from '@/application/types'
import { Entity, PrimaryKey, Property } from '@mikro-orm/core' import { Entity, PrimaryKey, Property } from '@mikro-orm/core'
import type { UUID } from '#application/types'
import { BaseEntity } from '#application/base/baseEntity'
export class BaseTile extends BaseEntity { export class BaseTile extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id = randomUUID() id = randomUUID()

View File

@ -1,11 +1,14 @@
import { randomUUID } from 'node:crypto' import { randomUUID } from 'node:crypto'
import { BaseEntity } from '@/application/base/baseEntity'
import type { UUID } from '@/application/types'
import { Character } from '@/entities/character'
import { PasswordResetToken } from '@/entities/passwordResetToken'
import { Collection, Entity, OneToMany, PrimaryKey, Property } from '@mikro-orm/core' import { Collection, Entity, OneToMany, PrimaryKey, Property } from '@mikro-orm/core'
import bcrypt from 'bcryptjs' import bcrypt from 'bcryptjs'
import type { UUID } from '#application/types'
import { BaseEntity } from '#application/base/baseEntity'
import { Character } from '#entities/character'
import { PasswordResetToken } from '#entities/passwordResetToken'
export class BaseUser extends BaseEntity { export class BaseUser extends BaseEntity {
@PrimaryKey() @PrimaryKey()
id = randomUUID() id = randomUUID()

View File

@ -1,6 +1,7 @@
import { BaseEntity } from '@/application/base/baseEntity'
import { Entity, PrimaryKey, Property } from '@mikro-orm/core' import { Entity, PrimaryKey, Property } from '@mikro-orm/core'
import { BaseEntity } from '#application/base/baseEntity'
export class BaseWorld extends BaseEntity { export class BaseWorld extends BaseEntity {
@PrimaryKey() @PrimaryKey()
date = new Date() date = new Date()

View File

@ -1,5 +1,6 @@
import { BaseCharacter } from '@/entities/base/character'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BaseCharacter } from '#entities/base/character'
@Entity() @Entity()
export class Character extends BaseCharacter {} export class Character extends BaseCharacter {}

View File

@ -1,5 +1,6 @@
import { BaseCharacterEquipment } from '@/entities/base/characterEquipment'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BaseCharacterEquipment } from '#entities/base/characterEquipment'
@Entity() @Entity()
export class CharacterEquipment extends BaseCharacterEquipment {} export class CharacterEquipment extends BaseCharacterEquipment {}

View File

@ -1,6 +1,7 @@
import { BaseCharacterHair } from '@/entities/base/characterHair'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BaseCharacterHair } from '#entities/base/characterHair'
@Entity() @Entity()
export class CharacterHair extends BaseCharacterHair { export class CharacterHair extends BaseCharacterHair {
public async cache() { public async cache() {

View File

@ -1,5 +1,6 @@
import { BaseCharacterItem } from '@/entities/base/characterItem'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BaseCharacterItem } from '#entities/base/characterItem'
@Entity() @Entity()
export class CharacterItem extends BaseCharacterItem {} export class CharacterItem extends BaseCharacterItem {}

View File

@ -1,6 +1,7 @@
import { BaseCharacterType } from '@/entities/base/characterType'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BaseCharacterType } from '#entities/base/characterType'
@Entity() @Entity()
export class CharacterType extends BaseCharacterType { export class CharacterType extends BaseCharacterType {
public async cache() { public async cache() {

View File

@ -1,5 +1,6 @@
import { BaseChat } from '@/entities/base/chat'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BaseChat } from '#entities/base/chat'
@Entity() @Entity()
export class Chat extends BaseChat {} export class Chat extends BaseChat {}

View File

@ -1,5 +1,6 @@
import { BaseItem } from '@/entities/base/item'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BaseItem } from '#entities/base/item'
@Entity() @Entity()
export class Item extends BaseItem {} export class Item extends BaseItem {}

View File

@ -1,8 +1,8 @@
import { BaseMap } from '@/entities/base/map'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BaseMap } from '#entities/base/map'
export type MapCacheT = ReturnType<Map['cache']> | {} export type MapCacheT = ReturnType<Map['cache']> | {}
export type MapEditorMapT = ReturnType<Map['mapEditorObject']> | {}
@Entity() @Entity()
export class Map extends BaseMap { export class Map extends BaseMap {
@ -36,42 +36,4 @@ export class Map extends BaseMap {
return {} return {}
} }
} }
public async mapEditorObject() {
try {
await this.getPlacedMapObjects().load()
await this.getMapEffects().load()
await this.getMapEventTiles().load()
return {
id: this.getId(),
name: this.getName(),
width: this.getWidth(),
height: this.getHeight(),
tiles: this.getTiles(),
pvp: this.getPvp(),
createdAt: this.getCreatedAt(),
updatedAt: this.getUpdatedAt(),
placedMapObjects: this.getPlacedMapObjects(),
mapEffects: this.getMapEffects(),
mapEventTiles: this.getMapEventTiles().map((mapEventTile) => ({
id: mapEventTile.getId(),
type: mapEventTile.getType(),
positionX: mapEventTile.getPositionX(),
positionY: mapEventTile.getPositionY(),
teleport: mapEventTile.getTeleport()
? {
toMap: mapEventTile.getTeleport()?.getToMap().getId(),
toPositionX: mapEventTile.getTeleport()?.getToPositionX(),
toPositionY: mapEventTile.getTeleport()?.getToPositionY(),
toRotation: mapEventTile.getTeleport()?.getToRotation()
}
: undefined
}))
}
} catch (error) {
console.error(error)
return {}
}
}
} }

View File

@ -1,5 +1,6 @@
import { BaseMapEffect } from '@/entities/base/mapEffect'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BaseMapEffect } from '#entities/base/mapEffect'
@Entity() @Entity()
export class MapEffect extends BaseMapEffect {} export class MapEffect extends BaseMapEffect {}

View File

@ -1,5 +1,6 @@
import { BaseMapEventTile } from '@/entities/base/mapEventTile'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BaseMapEventTile } from '#entities/base/mapEventTile'
@Entity() @Entity()
export class MapEventTile extends BaseMapEventTile {} export class MapEventTile extends BaseMapEventTile {}

View File

@ -1,5 +1,6 @@
import { BaseMapEventTileTeleport } from '@/entities/base/mapEventTileTeleport'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BaseMapEventTileTeleport } from '#entities/base/mapEventTileTeleport'
@Entity() @Entity()
export class MapEventTileTeleport extends BaseMapEventTileTeleport {} export class MapEventTileTeleport extends BaseMapEventTileTeleport {}

View File

@ -1,6 +1,7 @@
import { BaseMapObject } from '@/entities/base/mapObject'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BaseMapObject } from '#entities/base/mapObject'
@Entity() @Entity()
export class MapObject extends BaseMapObject { export class MapObject extends BaseMapObject {
public async cache() { public async cache() {

View File

@ -1,5 +1,6 @@
import { BasePasswordResetToken } from '@/entities/base/passwordResetToken'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BasePasswordResetToken } from '#entities/base/passwordResetToken'
@Entity() @Entity()
export class PasswordResetToken extends BasePasswordResetToken {} export class PasswordResetToken extends BasePasswordResetToken {}

View File

@ -1,5 +1,6 @@
import { BasePlacedMapObject } from '@/entities/base/placedMapObject'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BasePlacedMapObject } from '#entities/base/placedMapObject'
@Entity() @Entity()
export class PlacedMapObject extends BasePlacedMapObject {} export class PlacedMapObject extends BasePlacedMapObject {}

View File

@ -1,6 +1,7 @@
import { BaseSprite } from '@/entities/base/sprite'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BaseSprite } from '#entities/base/sprite'
@Entity() @Entity()
export class Sprite extends BaseSprite { export class Sprite extends BaseSprite {
public async cache() { public async cache() {

View File

@ -1,5 +1,6 @@
import { BaseSpriteAction } from '@/entities/base/spriteAction'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BaseSpriteAction } from '#entities/base/spriteAction'
@Entity() @Entity()
export class SpriteAction extends BaseSpriteAction {} export class SpriteAction extends BaseSpriteAction {}

View File

@ -1,6 +1,7 @@
import { BaseTile } from '@/entities/base/tile'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BaseTile } from '#entities/base/tile'
@Entity() @Entity()
export class Tile extends BaseTile { export class Tile extends BaseTile {
public async cache() { public async cache() {

View File

@ -1,5 +1,6 @@
import { BaseUser } from '@/entities/base/user'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BaseUser } from '#entities/base/user'
@Entity() @Entity()
export class User extends BaseUser {} export class User extends BaseUser {}

View File

@ -1,5 +1,6 @@
import { BaseWorld } from '@/entities/base/world'
import { Entity } from '@mikro-orm/core' import { Entity } from '@mikro-orm/core'
import { BaseWorld } from '#entities/base/world'
@Entity() @Entity()
export class World extends BaseWorld {} export class World extends BaseWorld {}

View File

@ -1,16 +1,14 @@
import { BaseEvent } from '@/application/base/baseEvent' import type { UUID } from '#application/types'
import { SocketEvent } from '@/application/enums'
import type { UUID } from '@/application/types' import { BaseEvent } from '#application/base/baseEvent'
import { ZCharacterConnect } from '@/application/zodTypes' import MapManager from '#managers/mapManager'
import MapManager from '@/managers/mapManager' import CharacterHairRepository from '#repositories/characterHairRepository'
import CharacterHairRepository from '@/repositories/characterHairRepository' import CharacterRepository from '#repositories/characterRepository'
import CharacterRepository from '@/repositories/characterRepository' import TeleportService from '#services/characterTeleportService'
import TeleportService from '@/services/characterTeleportService'
interface CharacterConnectPayload { interface CharacterConnectPayload {
characterId: UUID characterId: UUID
characterHairId: UUID | null characterHairId?: UUID
newNickname?: string
} }
export default class CharacterConnectEvent extends BaseEvent { export default class CharacterConnectEvent extends BaseEvent {
@ -18,40 +16,23 @@ export default class CharacterConnectEvent extends BaseEvent {
private readonly characterRepository = new CharacterRepository() private readonly characterRepository = new CharacterRepository()
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.CHARACTER_CONNECT, this.handleEvent.bind(this)) this.socket.on('character:connect', this.handleEvent.bind(this))
} }
private async handleEvent(data: CharacterConnectPayload, callback: (response: any) => void): Promise<void> { private async handleEvent(data: CharacterConnectPayload, callback: (response: any) => void): Promise<void> {
try { try {
if (data.newNickname === '') data.newNickname = undefined
const result = ZCharacterConnect.safeParse(data)
if (!result.success) {
this.sendNotificationAndLog(result.error?.errors[0]?.message ?? 'Invalid data')
return
}
if (await this.checkForActiveCharacters()) { if (await this.checkForActiveCharacters()) {
this.sendNotificationAndLog('You are already connected to another character') this.emitError('You are already connected to another character')
return return
} }
let character = await this.characterRepository.getByUserAndId(this.socket.userId!, data.characterId) const character = await this.characterRepository.getByUserAndId(this.socket.userId!, data.characterId)
if (!character) { if (!character) {
this.sendNotificationAndLog('Character not found or does not belong to this user') this.emitError('Character not found or does not belong to this user')
return return
} }
if (data.newNickname) {
const existingCharacter = await this.characterRepository.getByName(data.newNickname)
if (existingCharacter) {
this.sendNotificationAndLog('Nickname already in use: ' + data.newNickname)
return
}
await character.setName(data.newNickname).save()
}
// Set character id // Set character id
this.socket.characterId = character.id this.socket.characterId = character.id
@ -59,8 +40,6 @@ export default class CharacterConnectEvent extends BaseEvent {
if (data.characterHairId !== undefined && data.characterHairId !== null) { if (data.characterHairId !== undefined && data.characterHairId !== null) {
const characterHair = await this.characterHairRepository.getById(data.characterHairId) const characterHair = await this.characterHairRepository.getById(data.characterHairId)
await character.setCharacterHair(characterHair).save() await character.setCharacterHair(characterHair).save()
} else {
await character.setCharacterHair(null).save()
} }
// Emit character connect event // Emit character connect event

View File

@ -1,84 +1,72 @@
import { BaseEvent } from '@/application/base/baseEvent' import { ZodError } from 'zod'
import { SocketEvent } from '@/application/enums'
import { ZCharacterCreate } from '@/application/zodTypes'
import { Character } from '@/entities/character'
import CharacterRepository from '@/repositories/characterRepository'
import CharacterTypeRepository from '@/repositories/characterTypeRepository'
import MapRepository from '@/repositories/mapRepository'
import UserRepository from '@/repositories/userRepository'
import { z, ZodError } from 'zod'
const MAX_CHARACTERS = 4 import { BaseEvent } from '#application/base/baseEvent'
import { ZCharacterCreate } from '#application/zodTypes'
import { Character } from '#entities/character'
import CharacterRepository from '#repositories/characterRepository'
import CharacterTypeRepository from '#repositories/characterTypeRepository'
import MapRepository from '#repositories/mapRepository'
import UserRepository from '#repositories/userRepository'
export default class CharacterCreateEvent extends BaseEvent { export default class CharacterCreateEvent extends BaseEvent {
private readonly userRepository: UserRepository = new UserRepository()
private readonly characterRepository: CharacterRepository = new CharacterRepository()
private readonly characterTypeRepository: CharacterTypeRepository = new CharacterTypeRepository()
private readonly mapRepository: MapRepository = new MapRepository()
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.CHARACTER_CREATE, this.handleEvent.bind(this)) this.socket.on('character:create', this.handleEvent.bind(this))
} }
private async handleEvent(data: z.infer<typeof ZCharacterCreate>, callback: (success: boolean) => void): Promise<void> { private async handleEvent(data: any): Promise<any> {
// zod validate
try { try {
const validatedData = ZCharacterCreate.parse(data) data = ZCharacterCreate.parse(data)
await this.createCharacter(validatedData)
callback(true) const userRepository = new UserRepository()
} catch (error: unknown) { const characterRepository = new CharacterRepository()
this.returnError(error) const characterTypeRepository = new CharacterTypeRepository()
callback(false) const mapRepository = new MapRepository()
const user = await userRepository.getById(this.socket.userId!)
if (!user) {
return this.socket.emit('notification', { title: 'Error', message: 'You are not logged in' })
}
// Check if character name already exists
const characterExists = await characterRepository.getByName(data.name)
if (characterExists) {
return this.socket.emit('notification', { title: 'Error', message: 'Character name already exists' })
}
let characters: Character[] = await characterRepository.getByUserId(user.getId())
if (characters.length >= 4) {
return this.socket.emit('notification', { title: 'Error', message: 'You can only create 4 characters' })
}
// @TODO: Change to default location
const map = await mapRepository.getFirst()
// @TODO: Change to selected character type
const characterType = await characterTypeRepository.getFirst()
const newCharacter = new Character()
await newCharacter.setName(data.name).setUser(user).setMap(map!).setCharacterType(characterType).save()
if (!newCharacter) {
return this.socket.emit('notification', { title: 'Error', message: 'Failed to create character. Please try again (later).' })
}
characters = [...characters, newCharacter]
this.socket.emit('character:create:success')
this.socket.emit('character:list', characters)
this.logger.info('character:create success')
} catch (error: any) {
this.logger.error(`character:create error: ${error.message}`)
if (error instanceof ZodError) {
return this.socket.emit('notification', { title: 'Error', message: error.issues[0]!.message })
}
return this.socket.emit('notification', { title: 'Error', message: 'Could not create character. Please try again (later).' })
} }
} }
private async createCharacter(data: z.infer<typeof ZCharacterCreate>): Promise<void> {
const user = await this.userRepository.getById(this.socket.userId!)
if (!user) {
throw new Error('You are not logged in')
}
const characterExists = await this.characterRepository.getByName(data.name)
if (characterExists) {
throw new Error('Character name already exists')
}
let characters = await this.characterRepository.getByUserId(user.getId())
if (characters.length >= MAX_CHARACTERS) {
throw new Error(`You can only create ${MAX_CHARACTERS} characters`)
}
const map = await this.mapRepository.getFirst()
if (!map) {
throw new Error('No default map found')
}
const characterType = await this.characterTypeRepository.getFirst()
if (!characterType) {
throw new Error('No character type found')
}
const newCharacter = new Character()
await newCharacter.setName(data.name).setUser(user).setMap(map).setCharacterType(characterType).save()
characters = await this.characterRepository.getByUserId(user.getId())
this.socket.emit(SocketEvent.CHARACTER_LIST, characters)
this.logger.info('character:create success')
}
private returnError(error: unknown): void {
this.logger.error(`character:create error: ${error instanceof Error ? error.message : 'Unknown error'}`)
let errorMessage = 'Could not create character. Please try again later.'
if (error instanceof ZodError) {
errorMessage = error.issues[0]?.message || errorMessage
} else if (error instanceof Error) {
errorMessage = error.message
}
this.socket.emit(SocketEvent.NOTIFICATION, {
title: 'Error',
message: errorMessage
})
}
} }

View File

@ -1,8 +1,8 @@
import { BaseEvent } from '@/application/base/baseEvent' import type { UUID } from '#application/types'
import { SocketEvent } from '@/application/enums'
import type { UUID } from '@/application/types' import { BaseEvent } from '#application/base/baseEvent'
import { Character } from '@/entities/character' import { Character } from '#entities/character'
import CharacterRepository from '@/repositories/characterRepository' import CharacterRepository from '#repositories/characterRepository'
type TypePayload = { type TypePayload = {
characterId: UUID characterId: UUID
@ -14,18 +14,18 @@ type TypeResponse = {
export default class CharacterDeleteEvent extends BaseEvent { export default class CharacterDeleteEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.CHARACTER_DELETE, this.handleEvent.bind(this)) this.socket.on('character:delete', this.handleEvent.bind(this))
} }
private async handleEvent(data: TypePayload, callback: (response: TypeResponse) => void): Promise<any> { private async handleEvent(data: TypePayload, callback: (response: TypeResponse) => void): Promise<any> {
try { try {
const characterRepository = new CharacterRepository() const characterRepository = new CharacterRepository()
await (await characterRepository.getByUserAndId(this.socket.userId, data.characterId))?.delete() await (await characterRepository.getByUserAndId(this.socket.userId!, data.characterId))?.delete()
const characters: Character[] = await characterRepository.getByUserId(this.socket.userId) const characters: Character[] = await characterRepository.getByUserId(this.socket.userId!)
this.socket.emit(SocketEvent.CHARACTER_LIST, characters) this.socket.emit('character:list', characters)
} catch (error: any) { } catch (error: any) {
return this.socket.emit(SocketEvent.NOTIFICATION, { message: 'Character delete failed. Please try again.' }) return this.socket.emit('notification', { message: 'Character delete failed. Please try again.' })
} }
} }
} }

View File

@ -1,19 +1,18 @@
import { BaseEvent } from '@/application/base/baseEvent' import { BaseEvent } from '#application/base/baseEvent'
import { SocketEvent } from '@/application/enums' import { Character } from '#entities/character'
import { Character } from '@/entities/character' import CharacterRepository from '#repositories/characterRepository'
import CharacterRepository from '@/repositories/characterRepository'
export default class CharacterListEvent extends BaseEvent { export default class CharacterListEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.CHARACTER_LIST, this.handleEvent.bind(this)) this.socket.on('character:list', this.handleEvent.bind(this))
} }
private async handleEvent(data: any): Promise<void> { private async handleEvent(data: any): Promise<void> {
try { try {
const characterRepository = new CharacterRepository() const characterRepository = new CharacterRepository()
let characters: Character[] = await characterRepository.getByUserId(this.socket.userId) let characters: Character[] = await characterRepository.getByUserId(this.socket.userId!)
this.socket.emit(SocketEvent.CHARACTER_LIST, characters) this.socket.emit('character:list', characters)
} catch (error: any) { } catch (error: any) {
this.logger.error('character:list error', error.message) this.logger.error('character:list error', error.message)
} }

View File

@ -1,7 +1,6 @@
import { BaseEvent } from '@/application/base/baseEvent' import { BaseEvent } from '#application/base/baseEvent'
import { SocketEvent } from '@/application/enums' import CharacterRepository from '#repositories/characterRepository'
import CharacterRepository from '@/repositories/characterRepository' import ChatService from '#services/chatService'
import ChatService from '@/services/chatService'
type TypePayload = { type TypePayload = {
message: string message: string
@ -9,7 +8,7 @@ type TypePayload = {
export default class AlertCommandEvent extends BaseEvent { export default class AlertCommandEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.CHAT_MESSAGE, this.handleEvent.bind(this)) this.socket.on('chat:message', this.handleEvent.bind(this))
} }
private async handleEvent(data: TypePayload, callback: (response: boolean) => void): Promise<void> { private async handleEvent(data: TypePayload, callback: (response: boolean) => void): Promise<void> {
@ -26,7 +25,7 @@ export default class AlertCommandEvent extends BaseEvent {
return callback(false) return callback(false)
} }
this.io.emit(SocketEvent.NOTIFICATION, { title: 'Message from GM', message: args.join(' ') }) this.io.emit('notification', { title: 'Message from GM', message: args.join(' ') })
return callback(true) return callback(true)
} catch (error: any) { } catch (error: any) {
this.logger.error('chat:alert_command error', error.message) this.logger.error('chat:alert_command error', error.message)

View File

@ -1,8 +1,7 @@
import { BaseEvent } from '@/application/base/baseEvent' import { BaseEvent } from '#application/base/baseEvent'
import { SocketEvent } from '@/application/enums' import DateManager from '#managers/dateManager'
import DateManager from '@/managers/dateManager' import CharacterRepository from '#repositories/characterRepository'
import CharacterRepository from '@/repositories/characterRepository' import ChatService from '#services/chatService'
import ChatService from '@/services/chatService'
type TypePayload = { type TypePayload = {
message: string message: string
@ -10,7 +9,7 @@ type TypePayload = {
export default class SetTimeCommand extends BaseEvent { export default class SetTimeCommand extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.CHAT_MESSAGE, this.handleEvent.bind(this)) this.socket.on('chat:message', this.handleEvent.bind(this))
} }
private async handleEvent(data: TypePayload, callback: (response: boolean) => void): Promise<void> { private async handleEvent(data: TypePayload, callback: (response: boolean) => void): Promise<void> {

View File

@ -1,10 +1,10 @@
import { BaseEvent } from '@/application/base/baseEvent' import type { UUID } from '#application/types'
import { SocketEvent } from '@/application/enums'
import type { UUID } from '@/application/types' import { BaseEvent } from '#application/base/baseEvent'
import MapManager from '@/managers/mapManager' import MapManager from '#managers/mapManager'
import MapRepository from '@/repositories/mapRepository' import MapRepository from '#repositories/mapRepository'
import TeleportService from '@/services/characterTeleportService' import TeleportService from '#services/characterTeleportService'
import ChatService from '@/services/chatService' import ChatService from '#services/chatService'
type TypePayload = { type TypePayload = {
message: string message: string
@ -12,7 +12,7 @@ type TypePayload = {
export default class TeleportCommandEvent extends BaseEvent { export default class TeleportCommandEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.CHAT_MESSAGE, this.handleEvent.bind(this)) this.socket.on('chat:message', this.handleEvent.bind(this))
} }
private async handleEvent(data: TypePayload, callback: (response: boolean) => void) { private async handleEvent(data: TypePayload, callback: (response: boolean) => void) {
@ -29,7 +29,7 @@ export default class TeleportCommandEvent extends BaseEvent {
const args = ChatService.getArgs('teleport', data.message) const args = ChatService.getArgs('teleport', data.message)
if (!args || args.length === 0 || args.length > 3) { if (!args || args.length === 0 || args.length > 3) {
this.socket.emit(SocketEvent.NOTIFICATION, { this.socket.emit('notification', {
title: 'Server message', title: 'Server message',
message: 'Usage: /teleport <mapId> [x] [y]' message: 'Usage: /teleport <mapId> [x] [y]'
}) })
@ -41,7 +41,7 @@ export default class TeleportCommandEvent extends BaseEvent {
const targetY = args[2] ? parseInt(args[2], 10) : 0 const targetY = args[2] ? parseInt(args[2], 10) : 0
if (!mapId || isNaN(targetX) || isNaN(targetY)) { if (!mapId || isNaN(targetX) || isNaN(targetY)) {
this.socket.emit(SocketEvent.NOTIFICATION, { this.socket.emit('notification', {
title: 'Server message', title: 'Server message',
message: 'Invalid parameters. X and Y coordinates must be numbers.' message: 'Invalid parameters. X and Y coordinates must be numbers.'
}) })
@ -51,7 +51,7 @@ export default class TeleportCommandEvent extends BaseEvent {
const mapRepository = new MapRepository() const mapRepository = new MapRepository()
const map = await mapRepository.getById(mapId) const map = await mapRepository.getById(mapId)
if (!map) { if (!map) {
this.socket.emit(SocketEvent.NOTIFICATION, { this.socket.emit('notification', {
title: 'Server message', title: 'Server message',
message: 'Map not found' message: 'Map not found'
}) })
@ -59,7 +59,7 @@ export default class TeleportCommandEvent extends BaseEvent {
} }
if (character.map.id === map.id && targetX === character.positionX && targetY === character.positionY) { if (character.map.id === map.id && targetX === character.positionX && targetY === character.positionY) {
this.socket.emit(SocketEvent.NOTIFICATION, { this.socket.emit('notification', {
title: 'Server message', title: 'Server message',
message: 'You are already at that location' message: 'You are already at that location'
}) })
@ -74,20 +74,20 @@ export default class TeleportCommandEvent extends BaseEvent {
}) })
if (!success) { if (!success) {
return this.socket.emit(SocketEvent.NOTIFICATION, { return this.socket.emit('notification', {
title: 'Server message', title: 'Server message',
message: 'Failed to teleport' message: 'Failed to teleport'
}) })
} }
this.socket.emit(SocketEvent.NOTIFICATION, { this.socket.emit('notification', {
title: 'Server message', title: 'Server message',
message: `Teleported to ${map.name} (${targetX}, ${targetY})` message: `Teleported to ${map.name} (${targetX}, ${targetY})`
}) })
this.logger.info('teleport', `Character ${character.id} teleported to map ${map.id} at position (${targetX}, ${targetY})`) this.logger.info('teleport', `Character ${character.id} teleported to map ${map.id} at position (${targetX}, ${targetY})`)
} catch (error: any) { } catch (error: any) {
this.logger.error(`Error in teleport command: ${error.message}`) this.logger.error(`Error in teleport command: ${error.message}`)
this.socket.emit(SocketEvent.NOTIFICATION, { this.socket.emit('notification', {
title: 'Server message', title: 'Server message',
message: 'An error occurred while teleporting' message: 'An error occurred while teleporting'
}) })

View File

@ -1,7 +1,6 @@
import { BaseEvent } from '@/application/base/baseEvent' import { BaseEvent } from '#application/base/baseEvent'
import { SocketEvent } from '@/application/enums' import WeatherManager from '#managers/weatherManager'
import WeatherManager from '@/managers/weatherManager' import ChatService from '#services/chatService'
import ChatService from '@/services/chatService'
type TypePayload = { type TypePayload = {
message: string message: string
@ -9,7 +8,7 @@ type TypePayload = {
export default class ToggleFogCommand extends BaseEvent { export default class ToggleFogCommand extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.CHAT_MESSAGE, this.handleEvent.bind(this)) this.socket.on('chat:message', this.handleEvent.bind(this))
} }
private async handleEvent(data: TypePayload, callback: (response: boolean) => void): Promise<void> { private async handleEvent(data: TypePayload, callback: (response: boolean) => void): Promise<void> {

View File

@ -1,7 +1,6 @@
import { BaseEvent } from '@/application/base/baseEvent' import { BaseEvent } from '#application/base/baseEvent'
import { SocketEvent } from '@/application/enums' import WeatherManager from '#managers/weatherManager'
import WeatherManager from '@/managers/weatherManager' import ChatService from '#services/chatService'
import ChatService from '@/services/chatService'
type TypePayload = { type TypePayload = {
message: string message: string
@ -9,7 +8,7 @@ type TypePayload = {
export default class ToggleRainCommand extends BaseEvent { export default class ToggleRainCommand extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.CHAT_MESSAGE, this.handleEvent.bind(this)) this.socket.on('chat:message', this.handleEvent.bind(this))
} }
private async handleEvent(data: TypePayload, callback: (response: boolean) => void): Promise<void> { private async handleEvent(data: TypePayload, callback: (response: boolean) => void): Promise<void> {

View File

@ -1,7 +1,7 @@
import { BaseEvent } from '@/application/base/baseEvent' import { BaseEvent } from '#application/base/baseEvent'
import { SocketEvent } from '@/application/enums' import MapManager from '#managers/mapManager'
import MapManager from '@/managers/mapManager' import MapRepository from '#repositories/mapRepository'
import ChatService from '@/services/chatService' import ChatService from '#services/chatService'
type TypePayload = { type TypePayload = {
message: string message: string
@ -9,22 +9,31 @@ type TypePayload = {
export default class ChatMessageEvent extends BaseEvent { export default class ChatMessageEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.CHAT_MESSAGE, this.handleEvent.bind(this)) this.socket.on('chat:message', this.handleEvent.bind(this))
} }
private async handleEvent(data: TypePayload, callback: (response: boolean) => void): Promise<void> { private async handleEvent(data: TypePayload, callback: (response: boolean) => void): Promise<void> {
try { try {
if (!data.message || ChatService.isCommand(data.message)) return if (!data.message || ChatService.isCommand(data.message)) {
return callback(false)
}
const mapCharacter = MapManager.getCharacterById(this.socket.characterId!) const mapCharacter = MapManager.getCharacterById(this.socket.characterId!)
if (!mapCharacter) { if (!mapCharacter) {
this.logger.error('chat:message error', 'Character not found') this.logger.error('chat:message error', 'Character not found')
return return callback(false)
} }
const character = mapCharacter.character const character = mapCharacter.character
if (await ChatService.sendMapMessage(character.getId(), data.message)) { const mapRepository = new MapRepository()
const map = await mapRepository.getById(character.map.id)
if (!map) {
this.logger.error('chat:message error', 'Map not found')
return callback(false)
}
if (await ChatService.sendMapMessage(character.getId(), map.getId(), data.message)) {
return callback(true) return callback(true)
} }

View File

@ -1,10 +1,9 @@
import { BaseEvent } from '@/application/base/baseEvent' import { BaseEvent } from '#application/base/baseEvent'
import { SocketEvent } from '@/application/enums' import MapManager from '#managers/mapManager'
import MapManager from '@/managers/mapManager'
export default class DisconnectEvent extends BaseEvent { export default class DisconnectEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.DISCONNECT, this.handleEvent.bind(this)) this.socket.on('disconnect', this.handleEvent.bind(this))
} }
private async handleEvent(): Promise<void> { private async handleEvent(): Promise<void> {
@ -14,7 +13,7 @@ export default class DisconnectEvent extends BaseEvent {
return return
} }
this.io.emit(SocketEvent.USER_DISCONNECT, this.socket.userId) this.io.emit('user:disconnect', this.socket.userId)
const mapCharacter = MapManager.getCharacterById(this.socket.characterId!) const mapCharacter = MapManager.getCharacterById(this.socket.characterId!)
if (!mapCharacter) { if (!mapCharacter) {

View File

@ -1,28 +1,17 @@
import { BaseEvent } from '@/application/base/baseEvent' import { BaseEvent } from '#application/base/baseEvent'
import { CharacterGender, SocketEvent } from '@/application/enums' import { CharacterHair } from '#entities/characterHair'
import { CharacterHair } from '@/entities/characterHair'
import SpriteRepository from '@/repositories/spriteRepository'
export default class CharacterHairCreateEvent extends BaseEvent { export default class CharacterHairCreateEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.GM_CHARACTERHAIR_CREATE, this.handleEvent.bind(this)) this.socket.on('gm:characterHair:create', this.handleEvent.bind(this))
} }
private async handleEvent(data: undefined, callback: (response: boolean) => void): Promise<void> { private async handleEvent(data: undefined, callback: (response: boolean) => void): Promise<void> {
try { try {
if (!(await this.isCharacterGM())) return if (!(await this.isCharacterGM())) return
// Get first sprite
const spriteRepository = new SpriteRepository()
const firstSprite = await spriteRepository.getFirst()
if (!firstSprite) {
this.sendNotificationAndLog('No sprites found')
return callback(false)
}
const newCharacterHair = new CharacterHair() const newCharacterHair = new CharacterHair()
await newCharacterHair.setName('New hair').setGender(CharacterGender.MALE).setSprite(firstSprite).save() await newCharacterHair.setName('New hair').save()
return callback(true) return callback(true)
} catch (error) { } catch (error) {

View File

@ -1,7 +1,7 @@
import { BaseEvent } from '@/application/base/baseEvent' import type { UUID } from '#application/types'
import { SocketEvent } from '@/application/enums'
import type { UUID } from '@/application/types' import { BaseEvent } from '#application/base/baseEvent'
import CharacterHairRepository from '@/repositories/characterHairRepository' import CharacterHairRepository from '#repositories/characterHairRepository'
interface IPayload { interface IPayload {
id: UUID id: UUID
@ -9,7 +9,7 @@ interface IPayload {
export default class CharacterHairDeleteEvent extends BaseEvent { export default class CharacterHairDeleteEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.GM_CHARACTERHAIR_REMOVE, this.handleEvent.bind(this)) this.socket.on('gm:characterHair:remove', this.handleEvent.bind(this))
} }
private async handleEvent(data: IPayload, callback: (response: boolean) => void): Promise<void> { private async handleEvent(data: IPayload, callback: (response: boolean) => void): Promise<void> {

View File

@ -1,13 +1,12 @@
import { BaseEvent } from '@/application/base/baseEvent' import { BaseEvent } from '#application/base/baseEvent'
import { SocketEvent } from '@/application/enums' import { CharacterHair } from '#entities/characterHair'
import { CharacterHair } from '@/entities/characterHair' import CharacterHairRepository from '#repositories/characterHairRepository'
import CharacterHairRepository from '@/repositories/characterHairRepository'
interface IPayload {} interface IPayload {}
export default class characterHairListEvent extends BaseEvent { export default class characterHairListEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.GM_CHARACTERHAIR_LIST, this.handleEvent.bind(this)) this.socket.on('gm:characterHair:list', this.handleEvent.bind(this))
} }
private async handleEvent(data: IPayload, callback: (response: CharacterHair[]) => void): Promise<void> { private async handleEvent(data: IPayload, callback: (response: CharacterHair[]) => void): Promise<void> {

View File

@ -1,21 +1,21 @@
import { BaseEvent } from '@/application/base/baseEvent' import type { UUID } from '#application/types'
import { CharacterGender, SocketEvent } from '@/application/enums'
import type { UUID } from '@/application/types' import { BaseEvent } from '#application/base/baseEvent'
import CharacterHairRepository from '@/repositories/characterHairRepository' import { CharacterGender } from '#application/enums'
import SpriteRepository from '@/repositories/spriteRepository' import CharacterHairRepository from '#repositories/characterHairRepository'
import SpriteRepository from '#repositories/spriteRepository'
type Payload = { type Payload = {
id: UUID id: UUID
name: string name: string
gender: CharacterGender gender: CharacterGender
color: string
isSelectable: boolean isSelectable: boolean
spriteId: UUID spriteId: UUID
} }
export default class CharacterHairUpdateEvent extends BaseEvent { export default class CharacterHairUpdateEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.GM_CHARACTERHAIR_UPDATE, this.handleEvent.bind(this)) this.socket.on('gm:characterHair:update', this.handleEvent.bind(this))
} }
private async handleEvent(data: Payload, callback: (success: boolean) => void): Promise<void> { private async handleEvent(data: Payload, callback: (success: boolean) => void): Promise<void> {
@ -30,7 +30,7 @@ export default class CharacterHairUpdateEvent extends BaseEvent {
const characterHair = await characterHairRepository.getById(data.id) const characterHair = await characterHairRepository.getById(data.id)
if (!characterHair) return callback(false) if (!characterHair) return callback(false)
await characterHair.setName(data.name).setGender(data.gender).setColor(data.color).setIsSelectable(data.isSelectable).setSprite(sprite).setUpdatedAt(new Date()).save() await characterHair.setName(data.name).setGender(data.gender).setIsSelectable(data.isSelectable).setSprite(sprite).setUpdatedAt(new Date()).save()
return callback(true) return callback(true)
} catch (error) { } catch (error) {

View File

@ -1,10 +1,9 @@
import { BaseEvent } from '@/application/base/baseEvent' import { BaseEvent } from '#application/base/baseEvent'
import { SocketEvent } from '@/application/enums' import { CharacterType } from '#entities/characterType'
import { CharacterType } from '@/entities/characterType'
export default class CharacterTypeCreateEvent extends BaseEvent { export default class CharacterTypeCreateEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.GM_CHARACTERTYPE_CREATE, this.handleEvent.bind(this)) this.socket.on('gm:characterType:create', this.handleEvent.bind(this))
} }
private async handleEvent(data: undefined, callback: (response: boolean, characterType?: any) => void): Promise<void> { private async handleEvent(data: undefined, callback: (response: boolean, characterType?: any) => void): Promise<void> {

View File

@ -1,7 +1,7 @@
import { BaseEvent } from '@/application/base/baseEvent' import type { UUID } from '#application/types'
import { SocketEvent } from '@/application/enums'
import type { UUID } from '@/application/types' import { BaseEvent } from '#application/base/baseEvent'
import CharacterTypeRepository from '@/repositories/characterTypeRepository' import CharacterTypeRepository from '#repositories/characterTypeRepository'
interface IPayload { interface IPayload {
id: UUID id: UUID
@ -9,7 +9,7 @@ interface IPayload {
export default class CharacterTypeDeleteEvent extends BaseEvent { export default class CharacterTypeDeleteEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.GM_CHARACTERTYPE_REMOVE, this.handleEvent.bind(this)) this.socket.on('gm:characterType:remove', this.handleEvent.bind(this))
} }
private async handleEvent(data: IPayload, callback: (response: boolean) => void): Promise<void> { private async handleEvent(data: IPayload, callback: (response: boolean) => void): Promise<void> {

View File

@ -1,13 +1,12 @@
import { BaseEvent } from '@/application/base/baseEvent' import { BaseEvent } from '#application/base/baseEvent'
import { SocketEvent } from '@/application/enums' import { CharacterType } from '#entities/characterType'
import { CharacterType } from '@/entities/characterType' import CharacterTypeRepository from '#repositories/characterTypeRepository'
import CharacterTypeRepository from '@/repositories/characterTypeRepository'
interface IPayload {} interface IPayload {}
export default class CharacterTypeListEvent extends BaseEvent { export default class CharacterTypeListEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.GM_CHARACTERTYPE_LIST, this.handleEvent.bind(this)) this.socket.on('gm:characterType:list', this.handleEvent.bind(this))
} }
private async handleEvent(data: IPayload, callback: (response: CharacterType[]) => void): Promise<void> { private async handleEvent(data: IPayload, callback: (response: CharacterType[]) => void): Promise<void> {

View File

@ -1,8 +1,9 @@
import { BaseEvent } from '@/application/base/baseEvent' import type { UUID } from '#application/types'
import { CharacterGender, CharacterRace, SocketEvent } from '@/application/enums'
import type { UUID } from '@/application/types' import { BaseEvent } from '#application/base/baseEvent'
import CharacterTypeRepository from '@/repositories/characterTypeRepository' import { CharacterGender, CharacterRace } from '#application/enums'
import SpriteRepository from '@/repositories/spriteRepository' import CharacterTypeRepository from '#repositories/characterTypeRepository'
import SpriteRepository from '#repositories/spriteRepository'
type Payload = { type Payload = {
id: UUID id: UUID
@ -15,7 +16,7 @@ type Payload = {
export default class CharacterTypeUpdateEvent extends BaseEvent { export default class CharacterTypeUpdateEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.GM_CHARACTERTYPE_UPDATE, this.handleEvent.bind(this)) this.socket.on('gm:characterType:update', this.handleEvent.bind(this))
} }
private async handleEvent(data: Payload, callback: (success: boolean) => void): Promise<void> { private async handleEvent(data: Payload, callback: (success: boolean) => void): Promise<void> {

View File

@ -1,11 +1,11 @@
import { BaseEvent } from '@/application/base/baseEvent' import { BaseEvent } from '#application/base/baseEvent'
import { ItemRarity, ItemType, SocketEvent } from '@/application/enums' import { ItemRarity, ItemType } from '#application/enums'
import { Item } from '@/entities/item' import { Item } from '#entities/item'
import SpriteRepository from '@/repositories/spriteRepository' import SpriteRepository from '#repositories/spriteRepository'
export default class ItemCreateEvent extends BaseEvent { export default class ItemCreateEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.GM_ITEM_CREATE, this.handleEvent.bind(this)) this.socket.on('gm:item:create', this.handleEvent.bind(this))
} }
private async handleEvent(data: undefined, callback: (response: boolean, item?: any) => void): Promise<void> { private async handleEvent(data: undefined, callback: (response: boolean, item?: any) => void): Promise<void> {
@ -14,10 +14,7 @@ export default class ItemCreateEvent extends BaseEvent {
const spriteRepository = new SpriteRepository() const spriteRepository = new SpriteRepository()
const sprite = await spriteRepository.getFirst() const sprite = await spriteRepository.getFirst()
if (!sprite) { if (!sprite) return callback(false)
this.sendNotificationAndLog('No sprites found')
return callback(false)
}
const newItem = new Item() const newItem = new Item()
await newItem.setName('New Item').setItemType(ItemType.WEAPON).setStackable(false).setRarity(ItemRarity.COMMON).setSprite(sprite).save() await newItem.setName('New Item').setItemType(ItemType.WEAPON).setStackable(false).setRarity(ItemRarity.COMMON).setSprite(sprite).save()

View File

@ -1,7 +1,7 @@
import { BaseEvent } from '@/application/base/baseEvent' import type { UUID } from '#application/types'
import { SocketEvent } from '@/application/enums'
import type { UUID } from '@/application/types' import { BaseEvent } from '#application/base/baseEvent'
import ItemRepository from '@/repositories/itemRepository' import ItemRepository from '#repositories/itemRepository'
interface IPayload { interface IPayload {
id: UUID id: UUID
@ -9,7 +9,7 @@ interface IPayload {
export default class ItemDeleteEvent extends BaseEvent { export default class ItemDeleteEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.GM_ITEM_REMOVE, this.handleEvent.bind(this)) this.socket.on('gm:item:remove', this.handleEvent.bind(this))
} }
private async handleEvent(data: IPayload, callback: (response: boolean) => void): Promise<void> { private async handleEvent(data: IPayload, callback: (response: boolean) => void): Promise<void> {

View File

@ -1,13 +1,12 @@
import { BaseEvent } from '@/application/base/baseEvent' import { BaseEvent } from '#application/base/baseEvent'
import { SocketEvent } from '@/application/enums' import { Item } from '#entities/item'
import { Item } from '@/entities/item' import ItemRepository from '#repositories/itemRepository'
import ItemRepository from '@/repositories/itemRepository'
interface IPayload {} interface IPayload {}
export default class ItemListEvent extends BaseEvent { export default class ItemListEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.GM_ITEM_LIST, this.handleEvent.bind(this)) this.socket.on('gm:item:list', this.handleEvent.bind(this))
} }
private async handleEvent(data: IPayload, callback: (response: Item[]) => void): Promise<void> { private async handleEvent(data: IPayload, callback: (response: Item[]) => void): Promise<void> {

View File

@ -1,8 +1,9 @@
import { BaseEvent } from '@/application/base/baseEvent' import type { UUID } from '#application/types'
import { ItemRarity, ItemType, SocketEvent } from '@/application/enums'
import type { UUID } from '@/application/types' import { BaseEvent } from '#application/base/baseEvent'
import ItemRepository from '@/repositories/itemRepository' import { ItemType, ItemRarity } from '#application/enums'
import SpriteRepository from '@/repositories/spriteRepository' import ItemRepository from '#repositories/itemRepository'
import SpriteRepository from '#repositories/spriteRepository'
type Payload = { type Payload = {
id: UUID id: UUID
@ -16,7 +17,7 @@ type Payload = {
export default class ItemUpdateEvent extends BaseEvent { export default class ItemUpdateEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.GM_ITEM_UPDATE, this.handleEvent.bind(this)) this.socket.on('gm:item:update', this.handleEvent.bind(this))
} }
private async handleEvent(data: Payload, callback: (success: boolean) => void): Promise<void> { private async handleEvent(data: Payload, callback: (success: boolean) => void): Promise<void> {

View File

@ -1,13 +1,12 @@
import { BaseEvent } from '@/application/base/baseEvent' import { BaseEvent } from '#application/base/baseEvent'
import { SocketEvent } from '@/application/enums' import { MapObject } from '#entities/mapObject'
import { MapObject } from '@/entities/mapObject' import MapObjectRepository from '#repositories/mapObjectRepository'
import MapObjectRepository from '@/repositories/mapObjectRepository'
interface IPayload {} interface IPayload {}
export default class MapObjectListEvent extends BaseEvent { export default class MapObjectListEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.GM_MAPOBJECT_LIST, this.handleEvent.bind(this)) this.socket.on('gm:mapObject:list', this.handleEvent.bind(this))
} }
private async handleEvent(data: IPayload, callback: (response: MapObject[]) => void): Promise<void> { private async handleEvent(data: IPayload, callback: (response: MapObject[]) => void): Promise<void> {

View File

@ -1,9 +1,10 @@
import fs from 'fs' import fs from 'fs'
import { BaseEvent } from '@/application/base/baseEvent'
import { SocketEvent } from '@/application/enums' import type { UUID } from '#application/types'
import Storage from '@/application/storage'
import type { UUID } from '@/application/types' import { BaseEvent } from '#application/base/baseEvent'
import MapObjectRepository from '@/repositories/mapObjectRepository' import Storage from '#application/storage'
import MapObjectRepository from '#repositories/mapObjectRepository'
interface IPayload { interface IPayload {
mapObjectId: UUID mapObjectId: UUID
@ -11,7 +12,7 @@ interface IPayload {
export default class MapObjectRemoveEvent extends BaseEvent { export default class MapObjectRemoveEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.GM_MAPOBJECT_REMOVE, this.handleEvent.bind(this)) this.socket.on('gm:mapObject:remove', this.handleEvent.bind(this))
} }
private async handleEvent(data: IPayload, callback: (response: boolean) => void): Promise<void> { private async handleEvent(data: IPayload, callback: (response: boolean) => void): Promise<void> {

View File

@ -1,13 +1,12 @@
import { BaseEvent } from '@/application/base/baseEvent' import type { UUID } from '#application/types'
import { SocketEvent } from '@/application/enums'
import type { UUID } from '@/application/types' import { BaseEvent } from '#application/base/baseEvent'
import MapObjectRepository from '@/repositories/mapObjectRepository' import MapObjectRepository from '#repositories/mapObjectRepository'
type Payload = { type Payload = {
id: UUID id: UUID
name: string name: string
tags: string[] tags: string[]
depthOffsets: number[]
originX: number originX: number
originY: number originY: number
frameRate: number frameRate: number
@ -17,7 +16,7 @@ type Payload = {
export default class MapObjectUpdateEvent extends BaseEvent { export default class MapObjectUpdateEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.GM_MAPOBJECT_UPDATE, this.handleEvent.bind(this)) this.socket.on('gm:mapObject:update', this.handleEvent.bind(this))
} }
private async handleEvent(data: Payload, callback: (success: boolean) => void): Promise<void> { private async handleEvent(data: Payload, callback: (success: boolean) => void): Promise<void> {
@ -31,7 +30,6 @@ export default class MapObjectUpdateEvent extends BaseEvent {
if (data.name !== undefined) mapObject.name = data.name if (data.name !== undefined) mapObject.name = data.name
if (data.tags !== undefined) mapObject.tags = data.tags if (data.tags !== undefined) mapObject.tags = data.tags
if (data.depthOffsets !== undefined) mapObject.depthOffsets = data.depthOffsets
if (data.originX !== undefined) mapObject.originX = data.originX if (data.originX !== undefined) mapObject.originX = data.originX
if (data.originY !== undefined) mapObject.originY = data.originY if (data.originY !== undefined) mapObject.originY = data.originY
if (data.frameRate !== undefined) mapObject.frameRate = data.frameRate if (data.frameRate !== undefined) mapObject.frameRate = data.frameRate
@ -42,7 +40,7 @@ export default class MapObjectUpdateEvent extends BaseEvent {
return callback(true) return callback(true)
} catch (error) { } catch (error) {
this.socket.emit(SocketEvent.NOTIFICATION, { title: 'Error', message: 'Failed to update mapObject.' }) this.socket.emit('notification', { title: 'Error', message: 'Failed to update mapObject.' })
return callback(false) return callback(false)
} }
} }

View File

@ -1,18 +1,19 @@
import fs from 'fs/promises' import fs from 'fs/promises'
import { writeFile } from 'node:fs/promises' import { writeFile } from 'node:fs/promises'
import { BaseEvent } from '@/application/base/baseEvent'
import { SocketEvent } from '@/application/enums'
import Storage from '@/application/storage'
import { MapObject } from '@/entities/mapObject'
import sharp from 'sharp' import sharp from 'sharp'
import { BaseEvent } from '#application/base/baseEvent'
import Storage from '#application/storage'
import { MapObject } from '#entities/mapObject'
interface IObjectData { interface IObjectData {
[key: string]: Buffer [key: string]: Buffer
} }
export default class MapObjectUploadEvent extends BaseEvent { export default class MapObjectUploadEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.GM_MAPOBJECT_UPLOAD, this.handleEvent.bind(this)) this.socket.on('gm:mapObject:upload', this.handleEvent.bind(this))
} }
private async handleEvent(data: IObjectData, callback: (response: boolean) => void): Promise<void> { private async handleEvent(data: IObjectData, callback: (response: boolean) => void): Promise<void> {

View File

@ -1,9 +1,8 @@
import { BaseEvent } from '@/application/base/baseEvent' import type { UUID } from '#application/types'
import { SocketEvent } from '@/application/enums'
import type { UUID } from '@/application/types' import { BaseEvent } from '#application/base/baseEvent'
import { Sprite } from '@/entities/sprite' import { Sprite } from '#entities/sprite'
import { SpriteAction } from '@/entities/spriteAction' import SpriteRepository from '#repositories/spriteRepository'
import SpriteRepository from '@/repositories/spriteRepository'
interface CopyPayload { interface CopyPayload {
id: UUID id: UUID
@ -11,7 +10,7 @@ interface CopyPayload {
export default class SpriteCopyEvent extends BaseEvent { export default class SpriteCopyEvent extends BaseEvent {
public listen(): void { public listen(): void {
this.socket.on(SocketEvent.GM_SPRITE_COPY, this.handleEvent.bind(this)) this.socket.on('gm:sprite:copy', this.handleEvent.bind(this))
} }
private async handleEvent(payload: CopyPayload, callback: (success: boolean) => void): Promise<void> { private async handleEvent(payload: CopyPayload, callback: (success: boolean) => void): Promise<void> {
@ -30,21 +29,7 @@ export default class SpriteCopyEvent extends BaseEvent {
await spriteRepository.getEntityManager().populate(sourceSprite, ['spriteActions']) await spriteRepository.getEntityManager().populate(sourceSprite, ['spriteActions'])
const newSprite = new Sprite() const newSprite = new Sprite()
await newSprite.setName(`${sourceSprite.getName()} (Copy)`).save() await newSprite.setName(`${sourceSprite.getName()} (Copy)`).setSpriteActions(sourceSprite.getSpriteActions()).save()
for (const spriteAction of sourceSprite.getSpriteActions()) {
const newSpriteAction = new SpriteAction()
await newSpriteAction
.setSprite(newSprite)
.setAction(spriteAction.getAction())
.setSprites(spriteAction.getSprites() ?? [])
.setOriginX(spriteAction.getOriginX())
.setOriginY(spriteAction.getOriginY())
.setFrameWidth(spriteAction.getFrameWidth())
.setFrameHeight(spriteAction.getFrameHeight())
.setFrameRate(spriteAction.getFrameRate())
.save()
}
return callback(true) return callback(true)
} catch (error) { } catch (error) {

Some files were not shown because too many files have changed in this diff Show More