feat: Реализован полный CRUD для админ-панели и улучшена функциональность

- Portfolio CRUD: добавление, редактирование, удаление, переключение публикации
- Services CRUD: полное управление услугами с возможностью активации/деактивации
- Banner system: новая модель Banner с CRUD операциями и аналитикой кликов
- Telegram integration: расширенные настройки бота, обнаружение чатов, отправка сообщений
- Media management: улучшенная загрузка файлов с оптимизацией изображений и превью
- UI improvements: обновлённые админ-панели с rich-text редактором и drag&drop загрузкой
- Database: добавлена таблица banners с полями для баннеров и аналитики
This commit is contained in:
2025-10-22 20:32:16 +09:00
parent 150891b29d
commit 9477ff6de0
69 changed files with 11451 additions and 2321 deletions

View File

@@ -1,102 +1,96 @@
const mongoose = require('mongoose');
const { DataTypes } = require('sequelize');
const { sequelize } = require('../config/database');
const serviceSchema = new mongoose.Schema({
const Service = sequelize.define('Service', {
id: {
type: DataTypes.UUID,
defaultValue: DataTypes.UUIDV4,
primaryKey: true
},
name: {
type: String,
required: true,
trim: true
type: DataTypes.STRING,
allowNull: false,
validate: {
len: [1, 255]
},
set(value) {
this.setDataValue('name', value.trim());
}
},
description: {
type: String,
required: true
type: DataTypes.TEXT,
allowNull: false
},
shortDescription: {
type: String,
required: true,
maxlength: 150
type: DataTypes.STRING(150),
allowNull: false
},
icon: {
type: String,
required: true
type: DataTypes.STRING,
allowNull: false
},
category: {
type: String,
required: true,
enum: ['development', 'design', 'consulting', 'marketing', 'maintenance']
type: DataTypes.ENUM('development', 'design', 'consulting', 'marketing', 'maintenance'),
allowNull: false
},
features: {
type: DataTypes.JSONB,
defaultValue: []
},
features: [{
name: String,
description: String,
included: {
type: Boolean,
default: true
}
}],
pricing: {
basePrice: {
type: Number,
required: true,
min: 0
},
currency: {
type: String,
default: 'KRW'
},
priceType: {
type: String,
enum: ['fixed', 'hourly', 'project'],
default: 'project'
},
priceRange: {
min: Number,
max: Number
type: DataTypes.JSONB,
allowNull: false,
validate: {
isValidPricing(value) {
if (!value.basePrice || value.basePrice < 0) {
throw new Error('Base price must be a positive number');
}
}
}
},
estimatedTime: {
min: {
type: Number,
required: true
},
max: {
type: Number,
required: true
},
unit: {
type: String,
enum: ['hours', 'days', 'weeks', 'months'],
default: 'days'
type: DataTypes.JSONB,
allowNull: false,
validate: {
isValidTime(value) {
if (!value.min || !value.max || value.min > value.max) {
throw new Error('Invalid estimated time range');
}
}
}
},
isActive: {
type: Boolean,
default: true
type: DataTypes.BOOLEAN,
defaultValue: true
},
featured: {
type: Boolean,
default: false
type: DataTypes.BOOLEAN,
defaultValue: false
},
order: {
type: Number,
default: 0
type: DataTypes.INTEGER,
defaultValue: 0
},
tags: {
type: DataTypes.ARRAY(DataTypes.STRING),
defaultValue: []
},
portfolio: [{
type: mongoose.Schema.Types.ObjectId,
ref: 'Portfolio'
}],
tags: [{
type: String,
trim: true
}],
seo: {
metaTitle: String,
metaDescription: String,
keywords: [String]
type: DataTypes.JSONB,
defaultValue: {}
}
}, {
timestamps: true
tableName: 'services',
timestamps: true,
indexes: [
{
fields: ['category', 'featured', 'order']
},
{
type: 'gin',
fields: ['tags']
}
]
});
serviceSchema.index({ name: 'text', description: 'text', tags: 'text' });
serviceSchema.index({ category: 1, featured: -1, order: 1 });
module.exports = mongoose.model('Service', serviceSchema);
module.exports = Service;