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:
@@ -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;
|
||||
|
||||
@@ -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;
|
||||
|
||||
53
backend/models/match3/levelTileType.js
Normal file
53
backend/models/match3/levelTileType.js
Normal 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;
|
||||
@@ -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;
|
||||
|
||||
61
backend/models/match3/tileType.js
Normal file
61
backend/models/match3/tileType.js
Normal 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;
|
||||
@@ -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
|
||||
}
|
||||
]
|
||||
});
|
||||
|
||||
@@ -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
|
||||
}
|
||||
]
|
||||
});
|
||||
|
||||
Reference in New Issue
Block a user