feat(match3): Erweiterung der Match3-Admin-Funktionalitäten und -Modelle

- Implementierung neuer Endpunkte für die Verwaltung von Match3-Kampagnen, Levels, Objectives und Tile-Typen im Admin-Bereich.
- Anpassung der Admin-Services zur Unterstützung von Benutzerberechtigungen und Fehlerbehandlung.
- Einführung von neuen Modellen und Assoziationen für Match3-Levels und Tile-Typen in der Datenbank.
- Verbesserung der Internationalisierung für Match3-spezifische Texte in Deutsch und Englisch.
- Aktualisierung der Frontend-Routen und -Komponenten zur Verwaltung von Match3-Inhalten.
This commit is contained in:
Torsten Schulz (local)
2025-08-23 06:00:29 +02:00
parent 3eb7ae4e93
commit e168adeb51
40 changed files with 6474 additions and 1007 deletions

View File

@@ -34,7 +34,8 @@ const Campaign = sequelize.define('Campaign', {
}, {
tableName: 'match3_campaigns',
schema: 'match3',
timestamps: true
timestamps: true,
underscored: true // WICHTIG: Alle Datenbankfelder im snake_case Format
});
export default Campaign;

View File

@@ -1,7 +1,7 @@
import { sequelize } from '../../utils/sequelize.js';
import { DataTypes } from 'sequelize';
import { sequelize } from '../../utils/sequelize.js';
const Level = sequelize.define('Level', {
const Match3Level = sequelize.define('Match3Level', {
id: {
type: DataTypes.INTEGER,
primaryKey: true,
@@ -9,7 +9,11 @@ const Level = sequelize.define('Level', {
},
campaignId: {
type: DataTypes.INTEGER,
allowNull: false
allowNull: false,
references: {
model: 'match3_campaigns',
key: 'id'
}
},
name: {
type: DataTypes.STRING(255),
@@ -21,41 +25,51 @@ const Level = sequelize.define('Level', {
},
order: {
type: DataTypes.INTEGER,
defaultValue: 1
allowNull: false
},
boardSize: {
boardLayout: {
type: DataTypes.TEXT,
allowNull: true, // Ändern zu true, da bereits existierende Datensätze vorhanden sind
defaultValue: 'xxxxxx\nxxxxxx\nxxxxxx\nxxxxxx\nxxxxxx\nxxxxxx', // Standard-Layout für neue Level
comment: 'Level-Form als String (o = kein Feld, x = Feld, Zeilen durch \n getrennt)'
},
boardWidth: {
type: DataTypes.INTEGER,
defaultValue: 8
allowNull: true, // Ändern zu true, da bereits existierende Datensätze vorhanden sind
defaultValue: 6, // Standardwert für neue Level
comment: 'Breite des Level-Boards'
},
boardHeight: {
type: DataTypes.INTEGER,
allowNull: true, // Ändern zu true, da bereits existierende Datensätze vorhanden sind
defaultValue: 6, // Standardwert für neue Level
comment: 'Höhe des Level-Boards'
},
tileTypes: {
type: DataTypes.JSON,
allowNull: false,
defaultValue: ['gem', 'star', 'heart', 'diamond', 'circle', 'square']
allowNull: true, // Ändern zu true, da wir jetzt eine Verknüpfungstabelle haben
comment: 'Legacy: Array der verfügbaren Tile-Typen (wird durch levelTileTypes ersetzt)'
},
moveLimit: {
type: DataTypes.INTEGER,
allowNull: true
allowNull: false,
defaultValue: 20
},
timeLimit: {
type: DataTypes.INTEGER,
allowNull: true
allowNull: true,
comment: 'Zeitlimit in Sekunden (null = kein Limit)'
},
isActive: {
type: DataTypes.BOOLEAN,
allowNull: false,
defaultValue: true
},
createdAt: {
type: DataTypes.DATE,
defaultValue: DataTypes.NOW
},
updatedAt: {
type: DataTypes.DATE,
defaultValue: DataTypes.NOW
}
}, {
tableName: 'match3_levels',
schema: 'match3',
timestamps: true
timestamps: true,
underscored: true // WICHTIG: Alle Datenbankfelder im snake_case Format
});
export default Level;
export default Match3Level;

View File

@@ -0,0 +1,53 @@
import { DataTypes } from 'sequelize';
import { sequelize } from '../../utils/sequelize.js';
const Match3LevelTileType = sequelize.define('Match3LevelTileType', {
id: {
type: DataTypes.INTEGER,
primaryKey: true,
autoIncrement: true
},
levelId: {
type: DataTypes.INTEGER,
allowNull: false,
references: {
model: 'match3_levels',
key: 'id'
},
comment: 'Referenz auf den Level'
},
tileTypeId: {
type: DataTypes.INTEGER,
allowNull: false,
references: {
model: 'match3_tile_types',
key: 'id'
},
comment: 'Referenz auf den Tile-Typ'
},
weight: {
type: DataTypes.INTEGER,
allowNull: false,
defaultValue: 1,
comment: 'Gewichtung für die Wahrscheinlichkeit, dass dieser Tile-Typ erscheint'
},
isActive: {
type: DataTypes.BOOLEAN,
allowNull: false,
defaultValue: true,
comment: 'Ob dieser Tile-Typ in diesem Level aktiv ist'
}
}, {
tableName: 'match3_level_tile_types',
schema: 'match3',
timestamps: true,
underscored: true, // WICHTIG: Alle Datenbankfelder im snake_case Format
indexes: [
{
unique: true,
fields: ['level_id', 'tile_type_id'] // WICHTIG: Bei underscored: true müssen snake_case Namen verwendet werden
}
]
});
export default Match3LevelTileType;

View File

@@ -46,7 +46,8 @@ const Objective = sequelize.define('Objective', {
}, {
tableName: 'match3_objectives',
schema: 'match3',
timestamps: true
timestamps: true,
underscored: true // WICHTIG: Alle Datenbankfelder im snake_case Format
});
export default Objective;

View File

@@ -0,0 +1,61 @@
import { DataTypes } from 'sequelize';
import { sequelize } from '../../utils/sequelize.js';
const Match3TileType = sequelize.define('Match3TileType', {
id: {
type: DataTypes.INTEGER,
primaryKey: true,
autoIncrement: true
},
name: {
type: DataTypes.STRING(50),
allowNull: false,
comment: 'Eindeutiger Name des Tile-Typs (z.B. "gem", "star", "heart")'
},
displayName: {
type: DataTypes.STRING(100),
allowNull: false,
comment: 'Anzeigename des Tile-Typs (z.B. "Juwel", "Stern", "Herz")'
},
symbol: {
type: DataTypes.STRING(10),
allowNull: false,
comment: 'Unicode-Symbol für den Tile-Typ (z.B. "💎", "⭐", "❤️")'
},
color: {
type: DataTypes.STRING(7),
allowNull: true,
comment: 'Hex-Farbe für den Tile-Typ (z.B. "#ff6b6b")'
},
rarity: {
type: DataTypes.STRING(20),
allowNull: false,
defaultValue: 'common',
comment: 'Seltenheit des Tile-Typs (common, uncommon, rare, epic, legendary)'
},
points: {
type: DataTypes.INTEGER,
allowNull: false,
defaultValue: 10,
comment: 'Punkte, die dieser Tile-Typ beim Matchen gibt'
},
isActive: {
type: DataTypes.BOOLEAN,
allowNull: false,
defaultValue: true,
comment: 'Ob dieser Tile-Typ aktiv ist'
}
}, {
tableName: 'match3_tile_types',
schema: 'match3',
timestamps: true,
underscored: true, // WICHTIG: Alle Datenbankfelder im snake_case Format
indexes: [
{
unique: true,
fields: ['name']
}
]
});
export default Match3TileType;

View File

@@ -67,10 +67,11 @@ const UserLevelProgress = sequelize.define('UserLevelProgress', {
tableName: 'match3_user_level_progress',
schema: 'match3',
timestamps: true,
underscored: true, // WICHTIG: Alle Datenbankfelder im snake_case Format
indexes: [
{
unique: true,
fields: ['userProgressId', 'levelId']
fields: ['user_progress_id', 'level_id'] // WICHTIG: Bei underscored: true müssen snake_case Namen verwendet werden
}
]
});

View File

@@ -51,10 +51,11 @@ const UserProgress = sequelize.define('UserProgress', {
tableName: 'match3_user_progress',
schema: 'match3',
timestamps: true,
underscored: true, // WICHTIG: Alle Datenbankfelder im snake_case Format
indexes: [
{
unique: true,
fields: ['userId', 'campaignId']
fields: ['user_id', 'campaign_id'] // WICHTIG: Bei underscored: true müssen snake_case Namen verwendet werden
}
]
});