email and fix
This commit is contained in:
290
database.js
290
database.js
@@ -176,7 +176,174 @@ function createSQLiteTables() {
|
|||||||
)`);
|
)`);
|
||||||
|
|
||||||
console.log('✅ База данных SQLite инициализирована');
|
console.log('✅ База данных SQLite инициализирована');
|
||||||
setTimeout(addMissingColumns, 1000);
|
|
||||||
|
// Добавляем таблицу для пользовательских настроек
|
||||||
|
db.run(`CREATE TABLE IF NOT EXISTS user_settings (
|
||||||
|
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||||
|
user_id INTEGER UNIQUE NOT NULL,
|
||||||
|
email_notifications BOOLEAN DEFAULT true,
|
||||||
|
notification_email TEXT,
|
||||||
|
telegram_notifications BOOLEAN DEFAULT false,
|
||||||
|
telegram_chat_id TEXT,
|
||||||
|
vk_notifications BOOLEAN DEFAULT false,
|
||||||
|
vk_user_id TEXT,
|
||||||
|
created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
updated_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
FOREIGN KEY (user_id) REFERENCES users (id)
|
||||||
|
)`);
|
||||||
|
|
||||||
|
console.log('✅ Таблица для пользовательских настроек инициализирована');
|
||||||
|
|
||||||
|
// Запускаем проверку и обновление структуры таблиц
|
||||||
|
setTimeout(() => {
|
||||||
|
checkAndUpdateTableStructure();
|
||||||
|
}, 2000);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Функция для проверки и обновления структуры таблиц
|
||||||
|
function checkAndUpdateTableStructure() {
|
||||||
|
console.log('🔍 Проверка структуры таблиц...');
|
||||||
|
|
||||||
|
// Определяем ожидаемую структуру таблиц
|
||||||
|
const tableSchemas = {
|
||||||
|
users: [
|
||||||
|
{ name: 'id', type: 'INTEGER PRIMARY KEY AUTOINCREMENT' },
|
||||||
|
{ name: 'login', type: 'TEXT UNIQUE NOT NULL' },
|
||||||
|
{ name: 'password', type: 'TEXT' },
|
||||||
|
{ name: 'name', type: 'TEXT NOT NULL' },
|
||||||
|
{ name: 'email', type: 'TEXT UNIQUE NOT NULL' },
|
||||||
|
{ name: 'role', type: 'TEXT DEFAULT "teacher"' },
|
||||||
|
{ name: 'auth_type', type: 'TEXT DEFAULT "local"' },
|
||||||
|
{ name: 'groups', type: 'TEXT' },
|
||||||
|
{ name: 'description', type: 'TEXT' },
|
||||||
|
{ name: 'created_at', type: 'DATETIME DEFAULT CURRENT_TIMESTAMP' },
|
||||||
|
{ name: 'last_login', type: 'DATETIME' },
|
||||||
|
{ name: 'updated_at', type: 'DATETIME DEFAULT CURRENT_TIMESTAMP' }
|
||||||
|
],
|
||||||
|
tasks: [
|
||||||
|
{ name: 'id', type: 'INTEGER PRIMARY KEY AUTOINCREMENT' },
|
||||||
|
{ name: 'title', type: 'TEXT NOT NULL' },
|
||||||
|
{ name: 'description', type: 'TEXT' },
|
||||||
|
{ name: 'status', type: 'TEXT DEFAULT "active"' },
|
||||||
|
{ name: 'created_by', type: 'INTEGER NOT NULL' },
|
||||||
|
{ name: 'created_at', type: 'DATETIME DEFAULT CURRENT_TIMESTAMP' },
|
||||||
|
{ name: 'updated_at', type: 'DATETIME DEFAULT CURRENT_TIMESTAMP' },
|
||||||
|
{ name: 'deleted_at', type: 'DATETIME' },
|
||||||
|
{ name: 'deleted_by', type: 'INTEGER' },
|
||||||
|
{ name: 'original_task_id', type: 'INTEGER' },
|
||||||
|
{ name: 'start_date', type: 'DATETIME' },
|
||||||
|
{ name: 'due_date', type: 'DATETIME' },
|
||||||
|
{ name: 'rework_comment', type: 'TEXT' },
|
||||||
|
{ name: 'closed_at', type: 'DATETIME' },
|
||||||
|
{ name: 'closed_by', type: 'INTEGER' }
|
||||||
|
],
|
||||||
|
task_assignments: [
|
||||||
|
{ name: 'id', type: 'INTEGER PRIMARY KEY AUTOINCREMENT' },
|
||||||
|
{ name: 'task_id', type: 'INTEGER NOT NULL' },
|
||||||
|
{ name: 'user_id', type: 'INTEGER NOT NULL' },
|
||||||
|
{ name: 'status', type: 'TEXT DEFAULT "assigned"' },
|
||||||
|
{ name: 'start_date', type: 'DATETIME' },
|
||||||
|
{ name: 'due_date', type: 'DATETIME' },
|
||||||
|
{ name: 'rework_comment', type: 'TEXT' },
|
||||||
|
{ name: 'created_at', type: 'DATETIME DEFAULT CURRENT_TIMESTAMP' },
|
||||||
|
{ name: 'updated_at', type: 'DATETIME DEFAULT CURRENT_TIMESTAMP' }
|
||||||
|
],
|
||||||
|
task_files: [
|
||||||
|
{ name: 'id', type: 'INTEGER PRIMARY KEY AUTOINCREMENT' },
|
||||||
|
{ name: 'task_id', type: 'INTEGER NOT NULL' },
|
||||||
|
{ name: 'user_id', type: 'INTEGER NOT NULL' },
|
||||||
|
{ name: 'filename', type: 'TEXT NOT NULL' },
|
||||||
|
{ name: 'original_name', type: 'TEXT NOT NULL' },
|
||||||
|
{ name: 'file_path', type: 'TEXT NOT NULL' },
|
||||||
|
{ name: 'file_size', type: 'INTEGER NOT NULL' },
|
||||||
|
{ name: 'uploaded_at', type: 'DATETIME DEFAULT CURRENT_TIMESTAMP' }
|
||||||
|
],
|
||||||
|
activity_logs: [
|
||||||
|
{ name: 'id', type: 'INTEGER PRIMARY KEY AUTOINCREMENT' },
|
||||||
|
{ name: 'task_id', type: 'INTEGER NOT NULL' },
|
||||||
|
{ name: 'user_id', type: 'INTEGER NOT NULL' },
|
||||||
|
{ name: 'action', type: 'TEXT NOT NULL' },
|
||||||
|
{ name: 'details', type: 'TEXT' },
|
||||||
|
{ name: 'created_at', type: 'DATETIME DEFAULT CURRENT_TIMESTAMP' }
|
||||||
|
],
|
||||||
|
notification_logs: [
|
||||||
|
{ name: 'id', type: 'INTEGER PRIMARY KEY AUTOINCREMENT' },
|
||||||
|
{ name: 'notification_key', type: 'TEXT NOT NULL' },
|
||||||
|
{ name: 'created_at', type: 'DATETIME DEFAULT CURRENT_TIMESTAMP' }
|
||||||
|
],
|
||||||
|
user_settings: [
|
||||||
|
{ name: 'id', type: 'INTEGER PRIMARY KEY AUTOINCREMENT' },
|
||||||
|
{ name: 'user_id', type: 'INTEGER UNIQUE NOT NULL' },
|
||||||
|
{ name: 'email_notifications', type: 'BOOLEAN DEFAULT true' },
|
||||||
|
{ name: 'notification_email', type: 'TEXT' },
|
||||||
|
{ name: 'telegram_notifications', type: 'BOOLEAN DEFAULT false' },
|
||||||
|
{ name: 'telegram_chat_id', type: 'TEXT' },
|
||||||
|
{ name: 'vk_notifications', type: 'BOOLEAN DEFAULT false' },
|
||||||
|
{ name: 'vk_user_id', type: 'TEXT' },
|
||||||
|
{ name: 'created_at', type: 'DATETIME DEFAULT CURRENT_TIMESTAMP' },
|
||||||
|
{ name: 'updated_at', type: 'DATETIME DEFAULT CURRENT_TIMESTAMP' }
|
||||||
|
]
|
||||||
|
};
|
||||||
|
|
||||||
|
// Проверяем каждую таблицу
|
||||||
|
Object.entries(tableSchemas).forEach(([tableName, columns]) => {
|
||||||
|
db.all(`PRAGMA table_info(${tableName})`, (err, existingColumns) => {
|
||||||
|
if (err) {
|
||||||
|
console.error(`❌ Ошибка проверки таблицы ${tableName}:`, err.message);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (existingColumns.length === 0) {
|
||||||
|
console.log(`⚠️ Таблица ${tableName} не существует, создаем...`);
|
||||||
|
// Таблица будет создана автоматически при следующем запуске
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Создаем массив имен существующих колонок
|
||||||
|
const existingColumnNames = existingColumns.map(col => col.name.toLowerCase());
|
||||||
|
|
||||||
|
// Проверяем каждую ожидаемую колонку
|
||||||
|
columns.forEach(expectedColumn => {
|
||||||
|
const expectedName = expectedColumn.name.toLowerCase();
|
||||||
|
|
||||||
|
if (!existingColumnNames.includes(expectedName)) {
|
||||||
|
console.log(`🔧 Добавляем колонку ${expectedColumn.name} в таблицу ${tableName}...`);
|
||||||
|
|
||||||
|
db.run(
|
||||||
|
`ALTER TABLE ${tableName} ADD COLUMN ${expectedColumn.name} ${expectedColumn.type}`,
|
||||||
|
(alterErr) => {
|
||||||
|
if (alterErr) {
|
||||||
|
console.error(`❌ Ошибка добавления колонки ${expectedColumn.name}:`, alterErr.message);
|
||||||
|
} else {
|
||||||
|
console.log(`✅ Колонка ${expectedColumn.name} добавлена в таблицу ${tableName}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// Проверяем индекс для таблицы user_settings
|
||||||
|
setTimeout(() => {
|
||||||
|
db.get("SELECT name FROM sqlite_master WHERE type='index' AND name='idx_user_settings_user_id'", (err, index) => {
|
||||||
|
if (err) {
|
||||||
|
console.error('❌ Ошибка проверки индекса:', err.message);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!index) {
|
||||||
|
console.log('🔧 Создаем индекс для таблицы user_settings...');
|
||||||
|
db.run("CREATE INDEX IF NOT EXISTS idx_user_settings_user_id ON user_settings(user_id)", (createErr) => {
|
||||||
|
if (createErr) {
|
||||||
|
console.error('❌ Ошибка создания индекса:', createErr.message);
|
||||||
|
} else {
|
||||||
|
console.log('✅ Индекс для user_settings создан');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}, 1000);
|
||||||
}
|
}
|
||||||
|
|
||||||
function createPostgresAdapter(pool) {
|
function createPostgresAdapter(pool) {
|
||||||
@@ -400,6 +567,22 @@ async function createPostgresTables() {
|
|||||||
)
|
)
|
||||||
`);
|
`);
|
||||||
|
|
||||||
|
// Добавляем таблицу для пользовательских настроек
|
||||||
|
await client.query(`
|
||||||
|
CREATE TABLE IF NOT EXISTS user_settings (
|
||||||
|
id SERIAL PRIMARY KEY,
|
||||||
|
user_id INTEGER UNIQUE NOT NULL REFERENCES users(id),
|
||||||
|
email_notifications BOOLEAN DEFAULT true,
|
||||||
|
notification_email TEXT,
|
||||||
|
telegram_notifications BOOLEAN DEFAULT false,
|
||||||
|
telegram_chat_id TEXT,
|
||||||
|
vk_notifications BOOLEAN DEFAULT false,
|
||||||
|
vk_user_id TEXT,
|
||||||
|
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
||||||
|
)
|
||||||
|
`);
|
||||||
|
|
||||||
// Создаем индексы
|
// Создаем индексы
|
||||||
const indexes = [
|
const indexes = [
|
||||||
'CREATE INDEX IF NOT EXISTS idx_tasks_created_by ON tasks(created_by)',
|
'CREATE INDEX IF NOT EXISTS idx_tasks_created_by ON tasks(created_by)',
|
||||||
@@ -410,7 +593,8 @@ async function createPostgresTables() {
|
|||||||
'CREATE INDEX IF NOT EXISTS idx_task_assignments_status ON task_assignments(status)',
|
'CREATE INDEX IF NOT EXISTS idx_task_assignments_status ON task_assignments(status)',
|
||||||
'CREATE INDEX IF NOT EXISTS idx_task_files_task_id ON task_files(task_id)',
|
'CREATE INDEX IF NOT EXISTS idx_task_files_task_id ON task_files(task_id)',
|
||||||
'CREATE INDEX IF NOT EXISTS idx_activity_logs_task_id ON activity_logs(task_id)',
|
'CREATE INDEX IF NOT EXISTS idx_activity_logs_task_id ON activity_logs(task_id)',
|
||||||
'CREATE INDEX IF NOT EXISTS idx_activity_logs_created_at ON activity_logs(created_at)'
|
'CREATE INDEX IF NOT EXISTS idx_activity_logs_created_at ON activity_logs(created_at)',
|
||||||
|
'CREATE INDEX IF NOT EXISTS idx_user_settings_user_id ON user_settings(user_id)'
|
||||||
];
|
];
|
||||||
|
|
||||||
for (const indexQuery of indexes) {
|
for (const indexQuery of indexes) {
|
||||||
@@ -424,40 +608,91 @@ async function createPostgresTables() {
|
|||||||
client.release();
|
client.release();
|
||||||
console.log('✅ Таблицы PostgreSQL проверены/созданы');
|
console.log('✅ Таблицы PostgreSQL проверены/созданы');
|
||||||
|
|
||||||
|
// Проверяем структуру PostgreSQL таблиц
|
||||||
|
await checkPostgresTableStructure();
|
||||||
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('❌ Ошибка создания таблиц PostgreSQL:', error.message);
|
console.error('❌ Ошибка создания таблиц PostgreSQL:', error.message);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function addMissingColumns() {
|
// Функция для проверки структуры таблиц PostgreSQL
|
||||||
const columnsToAdd = [
|
async function checkPostgresTableStructure() {
|
||||||
{ table: 'tasks', column: 'rework_comment', type: 'TEXT' },
|
if (!USE_POSTGRES) return;
|
||||||
{ table: 'tasks', column: 'closed_at', type: 'DATETIME' },
|
|
||||||
{ table: 'tasks', column: 'closed_by', type: 'INTEGER' },
|
|
||||||
{ table: 'task_assignments', column: 'rework_comment', type: 'TEXT' }
|
|
||||||
];
|
|
||||||
|
|
||||||
columnsToAdd.forEach(({ table, column, type }) => {
|
try {
|
||||||
db.all(`PRAGMA table_info(${table})`, (err, rows) => {
|
const client = await postgresPool.connect();
|
||||||
if (err) {
|
|
||||||
console.error(`Ошибка при проверке таблицы ${table}:`, err);
|
console.log('🔍 Проверка структуры таблиц PostgreSQL...');
|
||||||
return;
|
|
||||||
|
// Определяем ожидаемую структуру таблиц PostgreSQL
|
||||||
|
const tableSchemas = {
|
||||||
|
user_settings: [
|
||||||
|
{ name: 'id', type: 'SERIAL PRIMARY KEY' },
|
||||||
|
{ name: 'user_id', type: 'INTEGER UNIQUE NOT NULL REFERENCES users(id)' },
|
||||||
|
{ name: 'email_notifications', type: 'BOOLEAN DEFAULT true' },
|
||||||
|
{ name: 'notification_email', type: 'TEXT' },
|
||||||
|
{ name: 'telegram_notifications', type: 'BOOLEAN DEFAULT false' },
|
||||||
|
{ name: 'telegram_chat_id', type: 'TEXT' },
|
||||||
|
{ name: 'vk_notifications', type: 'BOOLEAN DEFAULT false' },
|
||||||
|
{ name: 'vk_user_id', type: 'TEXT' },
|
||||||
|
{ name: 'created_at', type: 'TIMESTAMP DEFAULT CURRENT_TIMESTAMP' },
|
||||||
|
{ name: 'updated_at', type: 'TIMESTAMP DEFAULT CURRENT_TIMESTAMP' }
|
||||||
|
]
|
||||||
|
};
|
||||||
|
|
||||||
|
// Проверяем каждую таблицу
|
||||||
|
for (const [tableName, columns] of Object.entries(tableSchemas)) {
|
||||||
|
try {
|
||||||
|
// Проверяем существование таблицы
|
||||||
|
const tableExists = await client.query(
|
||||||
|
"SELECT EXISTS (SELECT FROM information_schema.tables WHERE table_name = $1)",
|
||||||
|
[tableName]
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!tableExists.rows[0].exists) {
|
||||||
|
console.log(`⚠️ Таблица ${tableName} не существует в PostgreSQL`);
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
const columnExists = rows.some(row => row.name === column);
|
// Получаем существующие колонки
|
||||||
if (!columnExists) {
|
const existingColumns = await client.query(`
|
||||||
db.run(`ALTER TABLE ${table} ADD COLUMN ${column} ${type}`, (err) => {
|
SELECT column_name, data_type, is_nullable, column_default
|
||||||
if (err) {
|
FROM information_schema.columns
|
||||||
console.error(`Ошибка при добавлении колонки ${column} в таблицу ${table}:`, err);
|
WHERE table_name = $1
|
||||||
} else {
|
ORDER BY ordinal_position
|
||||||
console.log(`✅ Добавлена колонка ${column} в таблицу ${table}`);
|
`, [tableName]);
|
||||||
|
|
||||||
|
const existingColumnNames = existingColumns.rows.map(col => col.column_name.toLowerCase());
|
||||||
|
|
||||||
|
// Проверяем каждую ожидаемую колонку
|
||||||
|
for (const expectedColumn of columns) {
|
||||||
|
const expectedName = expectedColumn.name.toLowerCase();
|
||||||
|
|
||||||
|
if (!existingColumnNames.includes(expectedName)) {
|
||||||
|
console.log(`🔧 Добавляем колонку ${expectedColumn.name} в таблицу PostgreSQL ${tableName}...`);
|
||||||
|
|
||||||
|
try {
|
||||||
|
await client.query(
|
||||||
|
`ALTER TABLE ${tableName} ADD COLUMN ${expectedColumn.name} ${expectedColumn.type}`
|
||||||
|
);
|
||||||
|
console.log(`✅ Колонка ${expectedColumn.name} добавлена в таблицу PostgreSQL ${tableName}`);
|
||||||
|
} catch (alterErr) {
|
||||||
|
console.error(`❌ Ошибка добавления колонки ${expectedColumn.name}:`, alterErr.message);
|
||||||
}
|
}
|
||||||
});
|
|
||||||
} else {
|
|
||||||
console.log(`✅ Колонка ${column} уже существует в таблице ${table}`);
|
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
});
|
} catch (error) {
|
||||||
|
console.error(`❌ Ошибка проверки таблицы PostgreSQL ${tableName}:`, error.message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
client.release();
|
||||||
|
console.log('✅ Проверка структуры таблиц PostgreSQL завершена');
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
console.error('❌ Ошибка проверки структуры таблиц PostgreSQL:', error.message);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function createTaskFolder(taskId) {
|
function createTaskFolder(taskId) {
|
||||||
@@ -598,7 +833,8 @@ module.exports = {
|
|||||||
updateTaskMetadata,
|
updateTaskMetadata,
|
||||||
checkTaskAccess,
|
checkTaskAccess,
|
||||||
USE_POSTGRES,
|
USE_POSTGRES,
|
||||||
getDatabaseType: () => USE_POSTGRES ? 'PostgreSQL' : 'SQLite'
|
getDatabaseType: () => USE_POSTGRES ? 'PostgreSQL' : 'SQLite',
|
||||||
|
checkAndUpdateTableStructure // Экспортируем для ручного запуска
|
||||||
};
|
};
|
||||||
|
|
||||||
// Запускаем инициализацию при экспорте (но она завершится позже)
|
// Запускаем инициализацию при экспорте (но она завершится позже)
|
||||||
|
|||||||
525
email-notifications.js
Normal file
525
email-notifications.js
Normal file
@@ -0,0 +1,525 @@
|
|||||||
|
// email-notifications.js
|
||||||
|
const nodemailer = require('nodemailer');
|
||||||
|
const { getDb } = require('./database');
|
||||||
|
|
||||||
|
class EmailNotifications {
|
||||||
|
constructor() {
|
||||||
|
this.transporter = null;
|
||||||
|
this.initialized = false;
|
||||||
|
this.init();
|
||||||
|
}
|
||||||
|
|
||||||
|
async init() {
|
||||||
|
try {
|
||||||
|
console.log('🔧 Инициализация Email уведомлений...');
|
||||||
|
|
||||||
|
if (!process.env.YANDEX_EMAIL || !process.env.YANDEX_PASSWORD) {
|
||||||
|
console.warn('⚠️ Настройки Яндекс почты не указаны в .env');
|
||||||
|
console.warn(' Email уведомления будут отключены');
|
||||||
|
this.initialized = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.transporter = nodemailer.createTransport({
|
||||||
|
host: process.env.YANDEX_SMTP_HOST || 'smtp.yandex.ru',
|
||||||
|
port: parseInt(process.env.YANDEX_SMTP_PORT) || 587,
|
||||||
|
secure: process.env.YANDEX_SMTP_SECURE === 'true',
|
||||||
|
auth: {
|
||||||
|
user: process.env.YANDEX_EMAIL,
|
||||||
|
pass: process.env.YANDEX_PASSWORD
|
||||||
|
},
|
||||||
|
tls: {
|
||||||
|
rejectUnauthorized: false
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Тестируем подключение
|
||||||
|
await this.transporter.verify();
|
||||||
|
this.initialized = true;
|
||||||
|
console.log('✅ Email уведомления инициализированы');
|
||||||
|
console.log(`📧 Отправитель: ${process.env.YANDEX_EMAIL}`);
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
console.error('❌ Ошибка инициализации Email уведомлений:', error.message);
|
||||||
|
this.initialized = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async getUserNotificationSettings(userId) {
|
||||||
|
if (!getDb) return null;
|
||||||
|
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
const db = getDb();
|
||||||
|
db.get(`
|
||||||
|
SELECT us.*, u.email as user_email, u.name as user_name
|
||||||
|
FROM user_settings us
|
||||||
|
LEFT JOIN users u ON us.user_id = u.id
|
||||||
|
WHERE us.user_id = ?
|
||||||
|
`, [userId], (err, settings) => {
|
||||||
|
if (err) {
|
||||||
|
reject(err);
|
||||||
|
} else {
|
||||||
|
resolve(settings);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async saveUserNotificationSettings(userId, settings) {
|
||||||
|
if (!getDb) return false;
|
||||||
|
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
const db = getDb();
|
||||||
|
const {
|
||||||
|
email_notifications = true,
|
||||||
|
notification_email = '',
|
||||||
|
telegram_notifications = false,
|
||||||
|
telegram_chat_id = '',
|
||||||
|
vk_notifications = false,
|
||||||
|
vk_user_id = ''
|
||||||
|
} = settings;
|
||||||
|
|
||||||
|
// Проверяем существование записи
|
||||||
|
db.get("SELECT id FROM user_settings WHERE user_id = ?", [userId], (err, existing) => {
|
||||||
|
if (err) {
|
||||||
|
reject(err);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (existing) {
|
||||||
|
// Обновляем существующую запись
|
||||||
|
db.run(`
|
||||||
|
UPDATE user_settings
|
||||||
|
SET email_notifications = ?,
|
||||||
|
notification_email = ?,
|
||||||
|
telegram_notifications = ?,
|
||||||
|
telegram_chat_id = ?,
|
||||||
|
vk_notifications = ?,
|
||||||
|
vk_user_id = ?,
|
||||||
|
updated_at = CURRENT_TIMESTAMP
|
||||||
|
WHERE user_id = ?
|
||||||
|
`, [
|
||||||
|
email_notifications ? 1 : 0,
|
||||||
|
notification_email,
|
||||||
|
telegram_notifications ? 1 : 0,
|
||||||
|
telegram_chat_id,
|
||||||
|
vk_notifications ? 1 : 0,
|
||||||
|
vk_user_id,
|
||||||
|
userId
|
||||||
|
], function(err) {
|
||||||
|
if (err) reject(err);
|
||||||
|
else resolve(true);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
// Создаем новую запись
|
||||||
|
db.run(`
|
||||||
|
INSERT INTO user_settings (
|
||||||
|
user_id, email_notifications, notification_email,
|
||||||
|
telegram_notifications, telegram_chat_id,
|
||||||
|
vk_notifications, vk_user_id
|
||||||
|
) VALUES (?, ?, ?, ?, ?, ?, ?)
|
||||||
|
`, [
|
||||||
|
userId,
|
||||||
|
email_notifications ? 1 : 0,
|
||||||
|
notification_email,
|
||||||
|
telegram_notifications ? 1 : 0,
|
||||||
|
telegram_chat_id,
|
||||||
|
vk_notifications ? 1 : 0,
|
||||||
|
vk_user_id
|
||||||
|
], function(err) {
|
||||||
|
if (err) reject(err);
|
||||||
|
else resolve(true);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async sendEmailNotification(to, subject, htmlContent) {
|
||||||
|
if (!this.initialized || !this.transporter) {
|
||||||
|
console.warn('⚠️ Email уведомления отключены');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const info = await this.transporter.sendMail({
|
||||||
|
from: `"School CRM" <${process.env.YANDEX_EMAIL}>`,
|
||||||
|
to: to,
|
||||||
|
subject: subject,
|
||||||
|
html: htmlContent,
|
||||||
|
text: htmlContent.replace(/<[^>]*>/g, '') // Конвертируем HTML в текст
|
||||||
|
});
|
||||||
|
|
||||||
|
console.log(`📧 Email отправлен: ${to}, Message ID: ${info.messageId}`);
|
||||||
|
return true;
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
console.error('❌ Ошибка отправки email:', error.message);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async sendTaskNotification(userId, taskData, notificationType) {
|
||||||
|
try {
|
||||||
|
const settings = await this.getUserNotificationSettings(userId);
|
||||||
|
if (!settings || !settings.email_notifications) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Используем указанную email или email из профиля пользователя
|
||||||
|
const emailTo = settings.notification_email || settings.user_email;
|
||||||
|
if (!emailTo) {
|
||||||
|
console.log(`⚠️ У пользователя ${userId} не указан email для уведомлений`);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
let subject = '';
|
||||||
|
let htmlContent = '';
|
||||||
|
|
||||||
|
switch (notificationType) {
|
||||||
|
case 'created':
|
||||||
|
subject = `Новая задача: ${taskData.title}`;
|
||||||
|
htmlContent = this.getTaskCreatedHtml(taskData);
|
||||||
|
break;
|
||||||
|
case 'updated':
|
||||||
|
subject = `Обновлена задача: ${taskData.title}`;
|
||||||
|
htmlContent = this.getTaskUpdatedHtml(taskData);
|
||||||
|
break;
|
||||||
|
case 'rework':
|
||||||
|
subject = `Задача возвращена на доработку: ${taskData.title}`;
|
||||||
|
htmlContent = this.getTaskReworkHtml(taskData);
|
||||||
|
break;
|
||||||
|
case 'closed':
|
||||||
|
subject = `Задача закрыта: ${taskData.title}`;
|
||||||
|
htmlContent = this.getTaskClosedHtml(taskData);
|
||||||
|
break;
|
||||||
|
case 'status_changed':
|
||||||
|
subject = `Изменен статус задачи: ${taskData.title}`;
|
||||||
|
htmlContent = this.getStatusChangedHtml(taskData);
|
||||||
|
break;
|
||||||
|
case 'deadline':
|
||||||
|
subject = `Скоро срок выполнения: ${taskData.title}`;
|
||||||
|
htmlContent = this.getDeadlineHtml(taskData);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
subject = `Уведомление по задаче: ${taskData.title}`;
|
||||||
|
htmlContent = this.getDefaultHtml(taskData);
|
||||||
|
}
|
||||||
|
|
||||||
|
return await this.sendEmailNotification(emailTo, subject, htmlContent);
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
console.error('❌ Ошибка отправки уведомления о задаче:', error);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HTML шаблоны для разных типов уведомлений
|
||||||
|
getTaskCreatedHtml(taskData) {
|
||||||
|
return `
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<style>
|
||||||
|
body { font-family: Arial, sans-serif; line-height: 1.6; }
|
||||||
|
.container { max-width: 600px; margin: 0 auto; padding: 20px; }
|
||||||
|
.header { background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); color: white; padding: 20px; border-radius: 10px 10px 0 0; }
|
||||||
|
.content { padding: 20px; border: 1px solid #ddd; border-top: none; }
|
||||||
|
.task-title { font-size: 18px; font-weight: bold; margin-bottom: 10px; }
|
||||||
|
.task-info { margin-bottom: 15px; }
|
||||||
|
.button { display: inline-block; background: #667eea; color: white; padding: 10px 20px; text-decoration: none; border-radius: 5px; }
|
||||||
|
.footer { margin-top: 20px; font-size: 12px; color: #666; }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div class="container">
|
||||||
|
<div class="header">
|
||||||
|
<h2>📋 Новая задача</h2>
|
||||||
|
</div>
|
||||||
|
<div class="content">
|
||||||
|
<div class="task-title">${taskData.title}</div>
|
||||||
|
<div class="task-info">
|
||||||
|
<p><strong>Описание:</strong> ${taskData.description || 'Без описания'}</p>
|
||||||
|
<p><strong>Срок выполнения:</strong> ${new Date(taskData.due_date).toLocaleString('ru-RU')}</p>
|
||||||
|
<p><strong>Создал:</strong> ${taskData.author_name || 'Неизвестно'}</p>
|
||||||
|
</div>
|
||||||
|
<p>Для просмотра подробной информации перейдите в систему управления задачами.</p>
|
||||||
|
<a href="${process.env.APP_URL || 'http://localhost:3000'}" class="button">Перейти в CRM</a>
|
||||||
|
</div>
|
||||||
|
<div class="footer">
|
||||||
|
<p>Это автоматическое уведомление от School CRM системы.</p>
|
||||||
|
<p>Вы можете изменить настройки уведомлений в личном кабинете.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
getTaskUpdatedHtml(taskData) {
|
||||||
|
return `
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<style>
|
||||||
|
body { font-family: Arial, sans-serif; line-height: 1.6; }
|
||||||
|
.container { max-width: 600px; margin: 0 auto; padding: 20px; }
|
||||||
|
.header { background: linear-gradient(135deg, #4CAF50 0%, #2E7D32 100%); color: white; padding: 20px; border-radius: 10px 10px 0 0; }
|
||||||
|
.content { padding: 20px; border: 1px solid #ddd; border-top: none; }
|
||||||
|
.task-title { font-size: 18px; font-weight: bold; margin-bottom: 10px; }
|
||||||
|
.task-info { margin-bottom: 15px; }
|
||||||
|
.button { display: inline-block; background: #4CAF50; color: white; padding: 10px 20px; text-decoration: none; border-radius: 5px; }
|
||||||
|
.footer { margin-top: 20px; font-size: 12px; color: #666; }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div class="container">
|
||||||
|
<div class="header">
|
||||||
|
<h2>🔄 Обновлена задача</h2>
|
||||||
|
</div>
|
||||||
|
<div class="content">
|
||||||
|
<div class="task-title">${taskData.title}</div>
|
||||||
|
<div class="task-info">
|
||||||
|
<p><strong>Изменения внес:</strong> ${taskData.author_name || 'Неизвестно'}</p>
|
||||||
|
<p><strong>Время:</strong> ${new Date().toLocaleString('ru-RU')}</p>
|
||||||
|
</div>
|
||||||
|
<p>Для просмотра изменений перейдите в систему управления задачами.</p>
|
||||||
|
<a href="${process.env.APP_URL || 'http://localhost:3000'}" class="button">Перейти в CRM</a>
|
||||||
|
</div>
|
||||||
|
<div class="footer">
|
||||||
|
<p>Это автоматическое уведомление от School CRM системы.</p>
|
||||||
|
<p>Вы можете изменить настройки уведомлений в личном кабинете.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
getTaskReworkHtml(taskData) {
|
||||||
|
return `
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<style>
|
||||||
|
body { font-family: Arial, sans-serif; line-height: 1.6; }
|
||||||
|
.container { max-width: 600px; margin: 0 auto; padding: 20px; }
|
||||||
|
.header { background: linear-gradient(135deg, #FF9800 0%, #F57C00 100%); color: white; padding: 20px; border-radius: 10px 10px 0 0; }
|
||||||
|
.content { padding: 20px; border: 1px solid #ddd; border-top: none; }
|
||||||
|
.task-title { font-size: 18px; font-weight: bold; margin-bottom: 10px; }
|
||||||
|
.task-info { margin-bottom: 15px; }
|
||||||
|
.comment-box { background: #FFF3E0; padding: 15px; border-left: 4px solid #FF9800; margin: 15px 0; }
|
||||||
|
.button { display: inline-block; background: #FF9800; color: white; padding: 10px 20px; text-decoration: none; border-radius: 5px; }
|
||||||
|
.footer { margin-top: 20px; font-size: 12px; color: #666; }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div class="container">
|
||||||
|
<div class="header">
|
||||||
|
<h2>🔄 Задача возвращена на доработку</h2>
|
||||||
|
</div>
|
||||||
|
<div class="content">
|
||||||
|
<div class="task-title">${taskData.title}</div>
|
||||||
|
<div class="task-info">
|
||||||
|
<p><strong>Автор замечания:</strong> ${taskData.author_name || 'Неизвестно'}</p>
|
||||||
|
</div>
|
||||||
|
<div class="comment-box">
|
||||||
|
<p><strong>Комментарий:</strong></p>
|
||||||
|
<p>${taskData.comment || 'Требуется доработка'}</p>
|
||||||
|
</div>
|
||||||
|
<p>Пожалуйста, исправьте замечания и обновите статус задачи.</p>
|
||||||
|
<a href="${process.env.APP_URL || 'http://localhost:3000'}" class="button">Перейти к задаче</a>
|
||||||
|
</div>
|
||||||
|
<div class="footer">
|
||||||
|
<p>Это автоматическое уведомление от School CRM системы.</p>
|
||||||
|
<p>Вы можете изменить настройки уведомлений в личном кабинете.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
getTaskClosedHtml(taskData) {
|
||||||
|
return `
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<style>
|
||||||
|
body { font-family: Arial, sans-serif; line-height: 1.6; }
|
||||||
|
.container { max-width: 600px; margin: 0 auto; padding: 20px; }
|
||||||
|
.header { background: linear-gradient(135deg, #9E9E9E 0%, #616161 100%); color: white; padding: 20px; border-radius: 10px 10px 0 0; }
|
||||||
|
.content { padding: 20px; border: 1px solid #ddd; border-top: none; }
|
||||||
|
.task-title { font-size: 18px; font-weight: bold; margin-bottom: 10px; }
|
||||||
|
.task-info { margin-bottom: 15px; }
|
||||||
|
.button { display: inline-block; background: #9E9E9E; color: white; padding: 10px 20px; text-decoration: none; border-radius: 5px; }
|
||||||
|
.footer { margin-top: 20px; font-size: 12px; color: #666; }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div class="container">
|
||||||
|
<div class="header">
|
||||||
|
<h2>✅ Задача закрыта</h2>
|
||||||
|
</div>
|
||||||
|
<div class="content">
|
||||||
|
<div class="task-title">${taskData.title}</div>
|
||||||
|
<div class="task-info">
|
||||||
|
<p><strong>Закрыта:</strong> ${taskData.author_name || 'Неизвестно'}</p>
|
||||||
|
<p><strong>Время закрытия:</strong> ${new Date().toLocaleString('ru-RU')}</p>
|
||||||
|
</div>
|
||||||
|
<p>Задача завершена и перемещена в архив.</p>
|
||||||
|
<a href="${process.env.APP_URL || 'http://localhost:3000'}" class="button">Перейти в CRM</a>
|
||||||
|
</div>
|
||||||
|
<div class="footer">
|
||||||
|
<p>Это автоматическое уведомление от School CRM системы.</p>
|
||||||
|
<p>Вы можете изменить настройки уведомлений в личном кабинете.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
getStatusChangedHtml(taskData) {
|
||||||
|
return `
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<style>
|
||||||
|
body { font-family: Arial, sans-serif; line-height: 1.6; }
|
||||||
|
.container { max-width: 600px; margin: 0 auto; padding: 20px; }
|
||||||
|
.header { background: linear-gradient(135deg, #2196F3 0%, #1976D2 100%); color: white; padding: 20px; border-radius: 10px 10px 0 0; }
|
||||||
|
.content { padding: 20px; border: 1px solid #ddd; border-top: none; }
|
||||||
|
.task-title { font-size: 18px; font-weight: bold; margin-bottom: 10px; }
|
||||||
|
.task-info { margin-bottom: 15px; }
|
||||||
|
.status-badge { display: inline-block; padding: 5px 10px; border-radius: 3px; color: white; }
|
||||||
|
.status-assigned { background: #FF9800; }
|
||||||
|
.status-in-progress { background: #2196F3; }
|
||||||
|
.status-completed { background: #4CAF50; }
|
||||||
|
.button { display: inline-block; background: #2196F3; color: white; padding: 10px 20px; text-decoration: none; border-radius: 5px; }
|
||||||
|
.footer { margin-top: 20px; font-size: 12px; color: #666; }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div class="container">
|
||||||
|
<div class="header">
|
||||||
|
<h2>🔄 Изменен статус задачи</h2>
|
||||||
|
</div>
|
||||||
|
<div class="content">
|
||||||
|
<div class="task-title">${taskData.title}</div>
|
||||||
|
<div class="task-info">
|
||||||
|
<p><strong>Новый статус:</strong> <span class="status-badge status-${taskData.status}">${this.getStatusText(taskData.status)}</span></p>
|
||||||
|
<p><strong>Изменил:</strong> ${taskData.user_name || taskData.author_name || 'Неизвестно'}</p>
|
||||||
|
<p><strong>Время:</strong> ${new Date().toLocaleString('ru-RU')}</p>
|
||||||
|
</div>
|
||||||
|
<p>Для просмотра деталей перейдите в систему управления задачами.</p>
|
||||||
|
<a href="${process.env.APP_URL || 'http://localhost:3000'}" class="button">Перейти в CRM</a>
|
||||||
|
</div>
|
||||||
|
<div class="footer">
|
||||||
|
<p>Это автоматическое уведомление от School CRM системы.</p>
|
||||||
|
<p>Вы можете изменить настройки уведомлений в личном кабинете.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
getDeadlineHtml(taskData) {
|
||||||
|
return `
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<style>
|
||||||
|
body { font-family: Arial, sans-serif; line-height: 1.6; }
|
||||||
|
.container { max-width: 600px; margin: 0 auto; padding: 20px; }
|
||||||
|
.header { background: linear-gradient(135deg, #F44336 0%, #D32F2F 100%); color: white; padding: 20px; border-radius: 10px 10px 0 0; }
|
||||||
|
.content { padding: 20px; border: 1px solid #ddd; border-top: none; }
|
||||||
|
.task-title { font-size: 18px; font-weight: bold; margin-bottom: 10px; }
|
||||||
|
.task-info { margin-bottom: 15px; }
|
||||||
|
.deadline-warning { background: #FFEBEE; padding: 15px; border-left: 4px solid #F44336; margin: 15px 0; }
|
||||||
|
.button { display: inline-block; background: #F44336; color: white; padding: 10px 20px; text-decoration: none; border-radius: 5px; }
|
||||||
|
.footer { margin-top: 20px; font-size: 12px; color: #666; }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div class="container">
|
||||||
|
<div class="header">
|
||||||
|
<h2>⚠️ Скоро срок выполнения</h2>
|
||||||
|
</div>
|
||||||
|
<div class="content">
|
||||||
|
<div class="task-title">${taskData.title}</div>
|
||||||
|
<div class="deadline-warning">
|
||||||
|
<p><strong>ВНИМАНИЕ!</strong> До окончания срока задачи осталось менее ${taskData.hours_left} часов!</p>
|
||||||
|
<p><strong>Срок выполнения:</strong> ${new Date(taskData.due_date).toLocaleString('ru-RU')}</p>
|
||||||
|
</div>
|
||||||
|
<p>Пожалуйста, завершите задачу в указанный срок.</p>
|
||||||
|
<a href="${process.env.APP_URL || 'http://localhost:3000'}" class="button">Перейти к задаче</a>
|
||||||
|
</div>
|
||||||
|
<div class="footer">
|
||||||
|
<p>Это автоматическое уведомление от School CRM системы.</p>
|
||||||
|
<p>Вы можете изменить настройки уведомлений в личном кабинете.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
getStatusText(status) {
|
||||||
|
const statusMap = {
|
||||||
|
'assigned': 'Назначена',
|
||||||
|
'in_progress': 'В работе',
|
||||||
|
'completed': 'Завершена',
|
||||||
|
'overdue': 'Просрочена',
|
||||||
|
'rework': 'На доработке'
|
||||||
|
};
|
||||||
|
return statusMap[status] || status;
|
||||||
|
}
|
||||||
|
|
||||||
|
getDefaultHtml(taskData) {
|
||||||
|
return `
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<style>
|
||||||
|
body { font-family: Arial, sans-serif; line-height: 1.6; }
|
||||||
|
.container { max-width: 600px; margin: 0 auto; padding: 20px; }
|
||||||
|
.header { background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); color: white; padding: 20px; border-radius: 10px 10px 0 0; }
|
||||||
|
.content { padding: 20px; border: 1px solid #ddd; border-top: none; }
|
||||||
|
.task-title { font-size: 18px; font-weight: bold; margin-bottom: 10px; }
|
||||||
|
.task-info { margin-bottom: 15px; }
|
||||||
|
.button { display: inline-block; background: #667eea; color: white; padding: 10px 20px; text-decoration: none; border-radius: 5px; }
|
||||||
|
.footer { margin-top: 20px; font-size: 12px; color: #666; }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div class="container">
|
||||||
|
<div class="header">
|
||||||
|
<h2>📢 Уведомление от School CRM</h2>
|
||||||
|
</div>
|
||||||
|
<div class="content">
|
||||||
|
<div class="task-title">${taskData.title}</div>
|
||||||
|
<div class="task-info">
|
||||||
|
<p>${taskData.message || 'Новое уведомление по задаче'}</p>
|
||||||
|
</div>
|
||||||
|
<a href="${process.env.APP_URL || 'http://localhost:3000'}" class="button">Перейти в CRM</a>
|
||||||
|
</div>
|
||||||
|
<div class="footer">
|
||||||
|
<p>Это автоматическое уведомление от School CRM системы.</p>
|
||||||
|
<p>Вы можете изменить настройки уведомлений в личном кабинете.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
isReady() {
|
||||||
|
return this.initialized;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Singleton
|
||||||
|
const emailNotifications = new EmailNotifications();
|
||||||
|
module.exports = emailNotifications;
|
||||||
417
notifications.js
417
notifications.js
@@ -1,97 +1,7 @@
|
|||||||
const fetch = require('node-fetch');
|
// notifications.js
|
||||||
const postgresLogger = require('./postgres');
|
const postgresLogger = require('./postgres');
|
||||||
const { getDb } = require('./database');
|
const { getDb } = require('./database');
|
||||||
|
const emailNotifications = require('./email-notifications');
|
||||||
async function sendDeadlineNotification(assignment, hoursLeft) {
|
|
||||||
try {
|
|
||||||
if (!process.env.NOTIFICATION_SERVICE_URL ||
|
|
||||||
!process.env.NOTIFICATION_SERVICE_LOGIN ||
|
|
||||||
!process.env.NOTIFICATION_SERVICE_PASSWORD) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const notificationKey = `deadline_${hoursLeft}h_task_${assignment.task_id}_user_${assignment.user_id}`;
|
|
||||||
const lastSent = await getLastNotificationSent(notificationKey);
|
|
||||||
const now = new Date();
|
|
||||||
|
|
||||||
if (lastSent) {
|
|
||||||
const timeSinceLast = now.getTime() - new Date(lastSent).getTime();
|
|
||||||
if (timeSinceLast < 12 * 60 * 60 * 1000) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const subject = `⚠️ До окончания срока задачи осталось менее ${hoursLeft} часов`;
|
|
||||||
const content = `Задача: ${assignment.title}\n\n` +
|
|
||||||
`Описание: ${assignment.description || 'Без описания'}\n` +
|
|
||||||
`Срок выполнения: ${new Date(assignment.due_date).toLocaleString('ru-RU')}\n` +
|
|
||||||
`Осталось времени: ${hoursLeft} часов\n\n` +
|
|
||||||
`Пожалуйста, завершите задачу в срок.`;
|
|
||||||
|
|
||||||
const recipients = [
|
|
||||||
{ id: assignment.user_id, name: assignment.user_name, email: assignment.user_email },
|
|
||||||
{ id: assignment.created_by, name: assignment.creator_name, email: assignment.creator_email }
|
|
||||||
].filter((value, index, self) =>
|
|
||||||
self.findIndex(r => r.id === value.id) === index
|
|
||||||
);
|
|
||||||
|
|
||||||
const recipientIds = recipients.map(r => r.id);
|
|
||||||
|
|
||||||
const authHeader = encodeBasicAuth(
|
|
||||||
process.env.NOTIFICATION_SERVICE_LOGIN,
|
|
||||||
process.env.NOTIFICATION_SERVICE_PASSWORD
|
|
||||||
);
|
|
||||||
|
|
||||||
const FormData = require('form-data');
|
|
||||||
const formData = new FormData();
|
|
||||||
formData.append('subject', subject);
|
|
||||||
formData.append('content', content);
|
|
||||||
formData.append('recipients', JSON.stringify(recipientIds));
|
|
||||||
formData.append('deliveryMethods', JSON.stringify(['email', 'telegram', 'vk']));
|
|
||||||
|
|
||||||
const response = await fetch(process.env.NOTIFICATION_SERVICE_URL, {
|
|
||||||
method: 'POST',
|
|
||||||
headers: {
|
|
||||||
'Authorization': `Basic ${authHeader}`
|
|
||||||
},
|
|
||||||
body: formData
|
|
||||||
});
|
|
||||||
|
|
||||||
if (response.ok) {
|
|
||||||
await saveNotificationSent(notificationKey);
|
|
||||||
console.log(`✅ Уведомление о сроке (${hoursLeft}ч) отправлено для задачи ${assignment.task_id}`);
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error('❌ Ошибка отправки уведомления о сроке:', error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function getLastNotificationSent(key) {
|
|
||||||
return new Promise((resolve) => {
|
|
||||||
const db = getDb();
|
|
||||||
if (!db) {
|
|
||||||
resolve(null);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
db.get("SELECT created_at FROM notification_logs WHERE notification_key = ? ORDER BY created_at DESC LIMIT 1",
|
|
||||||
[key], (err, row) => {
|
|
||||||
resolve(row ? row.created_at : null);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function saveNotificationSent(key) {
|
|
||||||
const db = getDb();
|
|
||||||
if (!db) return;
|
|
||||||
|
|
||||||
db.run("INSERT INTO notification_logs (notification_key) VALUES (?)", [key]);
|
|
||||||
}
|
|
||||||
|
|
||||||
function encodeBasicAuth(login, password) {
|
|
||||||
return Buffer.from(`${login}:${password}`).toString('base64');
|
|
||||||
}
|
|
||||||
|
|
||||||
async function sendTaskNotifications(type, taskId, taskTitle, taskDescription, authorId, comment = '', status = '', userName = '') {
|
async function sendTaskNotifications(type, taskId, taskTitle, taskDescription, authorId, comment = '', status = '', userName = '') {
|
||||||
try {
|
try {
|
||||||
@@ -101,31 +11,10 @@ async function sendTaskNotifications(type, taskId, taskTitle, taskDescription, a
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!process.env.NOTIFICATION_SERVICE_URL ||
|
|
||||||
!process.env.NOTIFICATION_SERVICE_LOGIN ||
|
|
||||||
!process.env.NOTIFICATION_SERVICE_PASSWORD) {
|
|
||||||
console.log('⚠️ Настройки сервиса уведомлений не заданы');
|
|
||||||
|
|
||||||
// Логируем в PostgreSQL даже если уведомления не отправляются
|
|
||||||
await logNotificationToPostgres({
|
|
||||||
type,
|
|
||||||
taskId,
|
|
||||||
taskTitle,
|
|
||||||
taskDescription,
|
|
||||||
authorId,
|
|
||||||
comment,
|
|
||||||
status,
|
|
||||||
userName,
|
|
||||||
error: 'Сервис уведомлений не настроен'
|
|
||||||
});
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`📢 Начинаем отправку уведомлений для задачи ${taskId}:`);
|
console.log(`📢 Начинаем отправку уведомлений для задачи ${taskId}:`);
|
||||||
console.log(` Тип: ${type}, Автор действия: ${authorId}, Название: ${taskTitle}`);
|
console.log(` Тип: ${type}, Автор действия: ${authorId}, Название: ${taskTitle}`);
|
||||||
|
|
||||||
// Получаем заказчика (создателя задачи) ОТДЕЛЬНО
|
// Получаем заказчика
|
||||||
const creator = await new Promise((resolve, reject) => {
|
const creator = await new Promise((resolve, reject) => {
|
||||||
db.get(`
|
db.get(`
|
||||||
SELECT t.created_by as user_id, u.name as user_name, u.login as user_login, u.email
|
SELECT t.created_by as user_id, u.name as user_name, u.login as user_login, u.email
|
||||||
@@ -138,7 +27,7 @@ async function sendTaskNotifications(type, taskId, taskTitle, taskDescription, a
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// Получаем исполнителей ОТДЕЛЬНО
|
// Получаем исполнителей
|
||||||
const assignees = await new Promise((resolve, reject) => {
|
const assignees = await new Promise((resolve, reject) => {
|
||||||
db.all(`
|
db.all(`
|
||||||
SELECT ta.user_id, u.name as user_name, u.login as user_login, u.email
|
SELECT ta.user_id, u.name as user_name, u.login as user_login, u.email
|
||||||
@@ -151,29 +40,9 @@ async function sendTaskNotifications(type, taskId, taskTitle, taskDescription, a
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// Собираем всех участников
|
|
||||||
const participants = [];
|
|
||||||
if (creator) {
|
|
||||||
participants.push({
|
|
||||||
...creator,
|
|
||||||
role: 'creator',
|
|
||||||
is_creator: true
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
if (assignees && assignees.length > 0) {
|
|
||||||
assignees.forEach(assignee => {
|
|
||||||
participants.push({
|
|
||||||
...assignee,
|
|
||||||
role: 'assignee',
|
|
||||||
is_creator: false
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// Получаем информацию об авторе действия
|
// Получаем информацию об авторе действия
|
||||||
const author = await new Promise((resolve, reject) => {
|
const author = await new Promise((resolve, reject) => {
|
||||||
db.get("SELECT name, login FROM users WHERE id = ?", [authorId], (err, row) => {
|
db.get("SELECT name, login, email FROM users WHERE id = ?", [authorId], (err, row) => {
|
||||||
if (err) reject(err);
|
if (err) reject(err);
|
||||||
else resolve(row);
|
else resolve(row);
|
||||||
});
|
});
|
||||||
@@ -182,8 +51,10 @@ async function sendTaskNotifications(type, taskId, taskTitle, taskDescription, a
|
|||||||
const authorName = author ? author.name : 'Система';
|
const authorName = author ? author.name : 'Система';
|
||||||
const authorLogin = author ? author.login : 'system';
|
const authorLogin = author ? author.login : 'system';
|
||||||
|
|
||||||
// Логируем в PostgreSQL
|
// Логируем в PostgreSQL (если настроено)
|
||||||
const postgresLogIds = await logNotificationToPostgres({
|
let postgresLogIds = [];
|
||||||
|
if (postgresLogger.initialized) {
|
||||||
|
postgresLogIds = await logNotificationToPostgres({
|
||||||
type,
|
type,
|
||||||
taskId,
|
taskId,
|
||||||
taskTitle,
|
taskTitle,
|
||||||
@@ -191,139 +62,123 @@ async function sendTaskNotifications(type, taskId, taskTitle, taskDescription, a
|
|||||||
authorId,
|
authorId,
|
||||||
authorName,
|
authorName,
|
||||||
authorLogin,
|
authorLogin,
|
||||||
participants,
|
participants: [...(creator ? [{...creator, role: 'creator'}] : []), ...assignees.map(a => ({...a, role: 'assignee'}))],
|
||||||
comment,
|
comment,
|
||||||
status,
|
status,
|
||||||
userName
|
userName
|
||||||
});
|
});
|
||||||
|
|
||||||
let subject, content;
|
|
||||||
|
|
||||||
switch (type) {
|
|
||||||
case 'created':
|
|
||||||
subject = `Новая задача: ${taskTitle}`;
|
|
||||||
content = `Создана новая задача:\n\n` +
|
|
||||||
`📋 ${taskTitle}\n` +
|
|
||||||
`📝 ${taskDescription || 'Без описания'}\n` +
|
|
||||||
`👤 Автор: ${authorName}\n\n` +
|
|
||||||
`Для просмотра перейдите в систему управления задачами.`;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 'updated':
|
|
||||||
subject = `Обновлена задача: ${taskTitle}`;
|
|
||||||
content = `Задача была обновлена:\n\n` +
|
|
||||||
`📋 ${taskTitle}\n` +
|
|
||||||
`📝 ${taskDescription || 'Без описания'}\n` +
|
|
||||||
`👤 Изменено: ${authorName}\n\n` +
|
|
||||||
`Для просмотра изменений перейдите в систему управления задачами.`;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 'rework':
|
|
||||||
subject = `Задача возвращена на доработку: ${taskTitle}`;
|
|
||||||
content = `Задача возвращена на доработку:\n\n` +
|
|
||||||
`📋 ${taskTitle}\n` +
|
|
||||||
`📝 Комментарий: ${comment}\n` +
|
|
||||||
`👤 Автор замечания: ${authorName}\n\n` +
|
|
||||||
`Пожалуйста, исправьте замечания и обновите статус задачи.`;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 'closed':
|
|
||||||
subject = `Задача закрыта: ${taskTitle}`;
|
|
||||||
content = `Задача была закрыта:\n\n` +
|
|
||||||
`📋 ${taskTitle}\n` +
|
|
||||||
`👤 Закрыта: ${authorName}\n\n` +
|
|
||||||
`Задача завершена и перемещена в архив.`;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 'status_changed':
|
|
||||||
const statusText = getStatusText(status);
|
|
||||||
subject = `Изменен статус задачи: ${taskTitle}`;
|
|
||||||
content = `Статус задачи изменен:\n\n` +
|
|
||||||
`📋 ${taskTitle}\n` +
|
|
||||||
`🔄 Новый статус: ${statusText}\n` +
|
|
||||||
`👤 Изменил: ${userName || authorName}\n\n` +
|
|
||||||
`Для просмотра перейдите в систему управления задачами.`;
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
console.log(`⚠️ Неизвестный тип уведомления: ${type}`);
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Фильтруем получателей: исключаем автора действия
|
// Отправляем email уведомления
|
||||||
const recipientIds = participants
|
const participants = [...(creator ? [creator] : []), ...assignees].filter(p => p.user_id !== authorId);
|
||||||
.filter(p => {
|
|
||||||
const shouldExclude = p.user_id === authorId;
|
|
||||||
if (shouldExclude) {
|
|
||||||
console.log(` ✋ Исключаем автора действия: ${p.user_name} (ID: ${p.user_id})`);
|
|
||||||
}
|
|
||||||
return !shouldExclude;
|
|
||||||
})
|
|
||||||
.map(p => p.user_id);
|
|
||||||
|
|
||||||
if (recipientIds.length === 0) {
|
for (const participant of participants) {
|
||||||
console.log('❌ Нет получателей для уведомления (все участники - автор изменения)');
|
const taskData = {
|
||||||
|
|
||||||
// Обновляем статус в PostgreSQL
|
|
||||||
await updatePostgresLogStatus(postgresLogIds, 'no_recipients', 'Нет получателей после фильтрации');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const authHeader = encodeBasicAuth(
|
|
||||||
process.env.NOTIFICATION_SERVICE_LOGIN,
|
|
||||||
process.env.NOTIFICATION_SERVICE_PASSWORD
|
|
||||||
);
|
|
||||||
|
|
||||||
const FormData = require('form-data');
|
|
||||||
const formData = new FormData();
|
|
||||||
formData.append('subject', subject);
|
|
||||||
formData.append('content', content);
|
|
||||||
formData.append('recipients', JSON.stringify(recipientIds));
|
|
||||||
formData.append('deliveryMethods', JSON.stringify(['email', 'telegram', 'vk']));
|
|
||||||
|
|
||||||
console.log(`🚀 Отправляем запрос на сервис уведомлений...`);
|
|
||||||
|
|
||||||
try {
|
|
||||||
const response = await fetch(process.env.NOTIFICATION_SERVICE_URL, {
|
|
||||||
method: 'POST',
|
|
||||||
headers: {
|
|
||||||
'Authorization': `Basic ${authHeader}`
|
|
||||||
},
|
|
||||||
body: formData
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!response.ok) {
|
|
||||||
throw new Error(`HTTP error! status: ${response.status}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
const result = await response.json();
|
|
||||||
console.log(`✅ Уведомления успешно отправлены для задачи ${taskId}`);
|
|
||||||
|
|
||||||
// Обновляем статус в PostgreSQL
|
|
||||||
await updatePostgresLogStatus(postgresLogIds, 'sent', null, new Date().toISOString());
|
|
||||||
|
|
||||||
console.log(` Результат от сервиса:`, result);
|
|
||||||
|
|
||||||
} catch (error) {
|
|
||||||
console.error('❌ Ошибка отправки уведомлений:', error);
|
|
||||||
|
|
||||||
// Обновляем статус в PostgreSQL
|
|
||||||
await updatePostgresLogStatus(postgresLogIds, 'failed', error.message);
|
|
||||||
|
|
||||||
console.error(' Детали ошибки:', {
|
|
||||||
taskId,
|
taskId,
|
||||||
type,
|
title: taskTitle,
|
||||||
authorId,
|
description: taskDescription,
|
||||||
errorMessage: error.message,
|
due_date: null, // Можно добавить получение срока из БД
|
||||||
stack: error.stack
|
author_name: authorName,
|
||||||
});
|
comment: comment,
|
||||||
|
status: status,
|
||||||
|
user_name: userName || participant.user_name,
|
||||||
|
hours_left: type === 'deadline' ? 24 : null // Для уведомлений о дедлайне
|
||||||
|
};
|
||||||
|
|
||||||
|
await emailNotifications.sendTaskNotification(
|
||||||
|
participant.user_id,
|
||||||
|
taskData,
|
||||||
|
type
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
console.log(`✅ Уведомления отправлены для задачи ${taskId}`);
|
||||||
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('❌ Общая ошибка при обработке уведомлений:', error);
|
console.error('❌ Общая ошибка при обработке уведомлений:', error);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Обновим функцию для уведомлений о дедлайнах
|
||||||
|
async function sendDeadlineNotification(assignment, hoursLeft) {
|
||||||
|
try {
|
||||||
|
const taskData = {
|
||||||
|
taskId: assignment.task_id,
|
||||||
|
title: assignment.title,
|
||||||
|
description: assignment.description || '',
|
||||||
|
due_date: assignment.due_date,
|
||||||
|
author_name: assignment.creator_name,
|
||||||
|
hours_left: hoursLeft
|
||||||
|
};
|
||||||
|
|
||||||
|
// Отправляем уведомление исполнителю
|
||||||
|
await emailNotifications.sendTaskNotification(
|
||||||
|
assignment.user_id,
|
||||||
|
taskData,
|
||||||
|
'deadline'
|
||||||
|
);
|
||||||
|
|
||||||
|
// Отправляем уведомление заказчику
|
||||||
|
await emailNotifications.sendTaskNotification(
|
||||||
|
assignment.created_by,
|
||||||
|
taskData,
|
||||||
|
'deadline'
|
||||||
|
);
|
||||||
|
|
||||||
|
console.log(`✅ Email уведомление о сроке (${hoursLeft}ч) отправлено для задачи ${assignment.task_id}`);
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
console.error('❌ Ошибка отправки email уведомления о сроке:', error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function checkUpcomingDeadlines() {
|
||||||
|
const now = new Date();
|
||||||
|
const in48Hours = new Date(now.getTime() + 48 * 60 * 60 * 1000).toISOString();
|
||||||
|
const in24Hours = new Date(now.getTime() + 24 * 60 * 60 * 1000).toISOString();
|
||||||
|
const nowISO = now.toISOString();
|
||||||
|
|
||||||
|
const query = `
|
||||||
|
SELECT DISTINCT ta.*, t.title, t.description, t.created_by, u.name as user_name, u.email as user_email,
|
||||||
|
creator.name as creator_name, creator.email as creator_email
|
||||||
|
FROM task_assignments ta
|
||||||
|
JOIN tasks t ON ta.task_id = t.id
|
||||||
|
JOIN users u ON ta.user_id = u.id
|
||||||
|
JOIN users creator ON t.created_by = creator.id
|
||||||
|
WHERE ta.due_date IS NOT NULL
|
||||||
|
AND ta.due_date > ?
|
||||||
|
AND ta.due_date <= ?
|
||||||
|
AND ta.status NOT IN ('completed', 'overdue')
|
||||||
|
AND t.status = 'active'
|
||||||
|
AND t.closed_at IS NULL
|
||||||
|
`;
|
||||||
|
|
||||||
|
const db = getDb();
|
||||||
|
if (!db) {
|
||||||
|
console.error('❌ База данных не доступна для проверки сроков');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
db.all(query, [nowISO, in48Hours], async (err, assignments) => {
|
||||||
|
if (err) {
|
||||||
|
console.error('❌ Ошибка при проверке сроков задач:', err);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const assignment of assignments) {
|
||||||
|
const dueDate = new Date(assignment.due_date);
|
||||||
|
const timeLeft = dueDate.getTime() - now.getTime();
|
||||||
|
const hoursLeft = Math.floor(timeLeft / (60 * 60 * 1000));
|
||||||
|
|
||||||
|
if (hoursLeft <= 48 && hoursLeft > 24) {
|
||||||
|
await sendDeadlineNotification(assignment, 48);
|
||||||
|
} else if (hoursLeft <= 24) {
|
||||||
|
await sendDeadlineNotification(assignment, 24);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
// Вспомогательные функции для работы с PostgreSQL
|
// Вспомогательные функции для работы с PostgreSQL
|
||||||
async function logNotificationToPostgres(data) {
|
async function logNotificationToPostgres(data) {
|
||||||
try {
|
try {
|
||||||
@@ -433,57 +288,11 @@ function getStatusText(status) {
|
|||||||
return statusMap[status] || status;
|
return statusMap[status] || status;
|
||||||
}
|
}
|
||||||
|
|
||||||
function checkUpcomingDeadlines() {
|
|
||||||
const now = new Date();
|
|
||||||
const in48Hours = new Date(now.getTime() + 48 * 60 * 60 * 1000).toISOString();
|
|
||||||
const in24Hours = new Date(now.getTime() + 24 * 60 * 60 * 1000).toISOString();
|
|
||||||
const nowISO = now.toISOString();
|
|
||||||
|
|
||||||
const query = `
|
|
||||||
SELECT DISTINCT ta.*, t.title, t.description, t.created_by, u.name as user_name, u.email as user_email,
|
|
||||||
creator.name as creator_name, creator.email as creator_email
|
|
||||||
FROM task_assignments ta
|
|
||||||
JOIN tasks t ON ta.task_id = t.id
|
|
||||||
JOIN users u ON ta.user_id = u.id
|
|
||||||
JOIN users creator ON t.created_by = creator.id
|
|
||||||
WHERE ta.due_date IS NOT NULL
|
|
||||||
AND ta.due_date > ?
|
|
||||||
AND ta.due_date <= ?
|
|
||||||
AND ta.status NOT IN ('completed', 'overdue')
|
|
||||||
AND t.status = 'active'
|
|
||||||
AND t.closed_at IS NULL
|
|
||||||
`;
|
|
||||||
|
|
||||||
const db = getDb();
|
|
||||||
if (!db) {
|
|
||||||
console.error('❌ База данных не доступна для проверки сроков');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
db.all(query, [nowISO, in48Hours], async (err, assignments) => {
|
|
||||||
if (err) {
|
|
||||||
console.error('❌ Ошибка при проверке сроков задач:', err);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const assignment of assignments) {
|
|
||||||
const dueDate = new Date(assignment.due_date);
|
|
||||||
const timeLeft = dueDate.getTime() - now.getTime();
|
|
||||||
const hoursLeft = Math.floor(timeLeft / (60 * 60 * 1000));
|
|
||||||
|
|
||||||
if (hoursLeft <= 48 && hoursLeft > 24) {
|
|
||||||
await sendDeadlineNotification(assignment, 48);
|
|
||||||
} else if (hoursLeft <= 24) {
|
|
||||||
await sendDeadlineNotification(assignment, 24);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// Экспортируем функции
|
// Экспортируем функции
|
||||||
module.exports = {
|
module.exports = {
|
||||||
sendTaskNotifications,
|
sendTaskNotifications,
|
||||||
checkUpcomingDeadlines,
|
checkUpcomingDeadlines,
|
||||||
sendDeadlineNotification,
|
sendDeadlineNotification,
|
||||||
getStatusText
|
getStatusText,
|
||||||
|
emailNotifications // Экспортируем для доступа к методам
|
||||||
};
|
};
|
||||||
@@ -15,6 +15,7 @@
|
|||||||
"form-data": "^4.0.0",
|
"form-data": "^4.0.0",
|
||||||
"multer": "^2.0.2",
|
"multer": "^2.0.2",
|
||||||
"node-fetch": "~2.6.7",
|
"node-fetch": "~2.6.7",
|
||||||
|
"nodemailer": "^6.9.13",
|
||||||
"pg": "^8.11.3",
|
"pg": "^8.11.3",
|
||||||
"sqlite3": "~5.1.6"
|
"sqlite3": "~5.1.6"
|
||||||
},
|
},
|
||||||
|
|||||||
97
public/auth.js
Normal file
97
public/auth.js
Normal file
@@ -0,0 +1,97 @@
|
|||||||
|
// auth.js - Аутентификация и авторизация
|
||||||
|
let currentUser = null;
|
||||||
|
|
||||||
|
async function checkAuth() {
|
||||||
|
try {
|
||||||
|
const response = await fetch('/api/user');
|
||||||
|
if (response.ok) {
|
||||||
|
const data = await response.json();
|
||||||
|
currentUser = data.user;
|
||||||
|
showMainInterface();
|
||||||
|
} else {
|
||||||
|
showLoginInterface();
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
showLoginInterface();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function showLoginInterface() {
|
||||||
|
document.getElementById('login-modal').style.display = 'block';
|
||||||
|
document.querySelector('.container').style.display = 'none';
|
||||||
|
}
|
||||||
|
|
||||||
|
function showMainInterface() {
|
||||||
|
document.getElementById('login-modal').style.display = 'none';
|
||||||
|
document.querySelector('.container').style.display = 'block';
|
||||||
|
|
||||||
|
let userInfo = `Вы вошли как: ${currentUser.name}`;
|
||||||
|
if (currentUser.auth_type === 'ldap') {
|
||||||
|
userInfo += ` (LDAP)`;
|
||||||
|
}
|
||||||
|
if (currentUser.groups && currentUser.groups.length > 0) {
|
||||||
|
userInfo += ` | Группы: ${currentUser.groups.join(', ')}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
document.getElementById('current-user').textContent = userInfo;
|
||||||
|
|
||||||
|
document.getElementById('tasks-controls').style.display = 'block';
|
||||||
|
|
||||||
|
const showDeletedLabel = document.querySelector('.show-deleted-label');
|
||||||
|
if (showDeletedLabel) {
|
||||||
|
if (currentUser.role === 'admin') {
|
||||||
|
showDeletedLabel.style.display = 'flex';
|
||||||
|
} else {
|
||||||
|
showDeletedLabel.style.display = 'none';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
loadUsers();
|
||||||
|
loadTasks();
|
||||||
|
loadActivityLogs();
|
||||||
|
showSection('tasks');
|
||||||
|
loadKanbanTasks();
|
||||||
|
|
||||||
|
showingTasksWithoutDate = false;
|
||||||
|
const btn = document.getElementById('tasks-no-date-btn');
|
||||||
|
if (btn) btn.classList.remove('active');
|
||||||
|
}
|
||||||
|
|
||||||
|
async function login(event) {
|
||||||
|
event.preventDefault();
|
||||||
|
|
||||||
|
const login = document.getElementById('login').value;
|
||||||
|
const password = document.getElementById('password').value;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch('/api/login', {
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
body: JSON.stringify({ login, password })
|
||||||
|
});
|
||||||
|
|
||||||
|
if (response.ok) {
|
||||||
|
const data = await response.json();
|
||||||
|
currentUser = data.user;
|
||||||
|
showMainInterface();
|
||||||
|
} else {
|
||||||
|
const error = await response.json();
|
||||||
|
alert(error.error || 'Ошибка входа');
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка:', error);
|
||||||
|
alert('Ошибка подключения к серверу');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function logout() {
|
||||||
|
try {
|
||||||
|
await fetch('/api/logout', { method: 'POST' });
|
||||||
|
currentUser = null;
|
||||||
|
showLoginInterface();
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка выхода:', error);
|
||||||
|
}
|
||||||
|
}
|
||||||
451
public/files.js
Normal file
451
public/files.js
Normal file
@@ -0,0 +1,451 @@
|
|||||||
|
// files.js - Работа с файлами
|
||||||
|
let currentTaskFiles = [];
|
||||||
|
let currentEditTaskFiles = [];
|
||||||
|
|
||||||
|
function initializeFileUploads() {
|
||||||
|
// Создание задачи
|
||||||
|
document.getElementById('files').addEventListener('change', function(e) {
|
||||||
|
currentTaskFiles = Array.from(e.target.files);
|
||||||
|
updateFileList();
|
||||||
|
});
|
||||||
|
|
||||||
|
// Редактирование задачи
|
||||||
|
document.getElementById('edit-files').addEventListener('change', function(e) {
|
||||||
|
const newFiles = Array.from(e.target.files);
|
||||||
|
currentEditTaskFiles.push(...newFiles);
|
||||||
|
updateEditFileList();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function updateFileList() {
|
||||||
|
const fileInput = document.getElementById('files');
|
||||||
|
const fileList = document.getElementById('file-list');
|
||||||
|
updateFileListForInput(fileInput, fileList);
|
||||||
|
}
|
||||||
|
|
||||||
|
function updateEditFileList() {
|
||||||
|
const fileInput = document.getElementById('edit-files');
|
||||||
|
const fileList = document.getElementById('edit-file-list');
|
||||||
|
|
||||||
|
// Используем улучшенный рендеринг файлов
|
||||||
|
const files = fileInput.files;
|
||||||
|
const existingFiles = currentEditTaskFiles.filter(file => !(file instanceof File));
|
||||||
|
|
||||||
|
if (files.length === 0 && existingFiles.length === 0) {
|
||||||
|
fileList.innerHTML = '';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let html = '<ul>';
|
||||||
|
let totalSize = 0;
|
||||||
|
|
||||||
|
// Существующие файлы
|
||||||
|
existingFiles.forEach(file => {
|
||||||
|
totalSize += file.file_size;
|
||||||
|
html += `<li>${file.original_name} (${(file.file_size / 1024 / 1024).toFixed(2)} MB) - <em>уже загружен</em></li>`;
|
||||||
|
});
|
||||||
|
|
||||||
|
// Новые файлы
|
||||||
|
for (let i = 0; i < files.length; i++) {
|
||||||
|
const file = files[i];
|
||||||
|
totalSize += file.size;
|
||||||
|
html += `<li>${file.name} (${(file.size / 1024 / 1024).toFixed(2)} MB) - <em>новый</em></li>`;
|
||||||
|
}
|
||||||
|
|
||||||
|
html += '</ul>';
|
||||||
|
html += `<p><strong>Общий размер: ${(totalSize / 1024 / 1024).toFixed(2)} MB / 300 MB</strong></p>`;
|
||||||
|
|
||||||
|
fileList.innerHTML = html;
|
||||||
|
}
|
||||||
|
|
||||||
|
function updateFileListForInput(fileInput, fileList) {
|
||||||
|
const files = fileInput.files;
|
||||||
|
|
||||||
|
if (files.length === 0) {
|
||||||
|
fileList.innerHTML = '';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let html = '<ul>';
|
||||||
|
let totalSize = 0;
|
||||||
|
|
||||||
|
for (let i = 0; i < files.length; i++) {
|
||||||
|
const file = files[i];
|
||||||
|
totalSize += file.size;
|
||||||
|
html += `<li>${file.name} (${(file.size / 1024 / 1024).toFixed(2)} MB)</li>`;
|
||||||
|
}
|
||||||
|
|
||||||
|
html += '</ul>';
|
||||||
|
html += `<p><strong>Общий размер: ${(totalSize / 1024 / 1024).toFixed(2)} MB / 300 MB</strong></p>`;
|
||||||
|
|
||||||
|
fileList.innerHTML = html;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Удаление файлов из списка
|
||||||
|
function removeFile(index) {
|
||||||
|
currentTaskFiles.splice(index, 1);
|
||||||
|
updateFileList();
|
||||||
|
|
||||||
|
// Обновляем input files
|
||||||
|
const dataTransfer = new DataTransfer();
|
||||||
|
currentTaskFiles.forEach(file => dataTransfer.items.add(file));
|
||||||
|
document.getElementById('files').files = dataTransfer.files;
|
||||||
|
}
|
||||||
|
|
||||||
|
function removeEditFile(index) {
|
||||||
|
currentEditTaskFiles.splice(index, 1);
|
||||||
|
updateEditFileList();
|
||||||
|
|
||||||
|
// Обновляем input files
|
||||||
|
const dataTransfer = new DataTransfer();
|
||||||
|
const newFiles = currentEditTaskFiles.filter(file => file instanceof File);
|
||||||
|
newFiles.forEach(file => dataTransfer.items.add(file));
|
||||||
|
document.getElementById('edit-files').files = dataTransfer.files;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function loadTaskFiles(taskId) {
|
||||||
|
try {
|
||||||
|
const response = await fetch(`/api/tasks/${taskId}/files`);
|
||||||
|
const files = await response.json();
|
||||||
|
|
||||||
|
const container = document.getElementById(`files-${taskId}`);
|
||||||
|
if (container) {
|
||||||
|
if (files.length === 0) {
|
||||||
|
container.innerHTML = '<strong>Файлы:</strong> <span class="files-placeholder">скрыто</span>';
|
||||||
|
} else {
|
||||||
|
container.innerHTML = `
|
||||||
|
<strong>Файлы:</strong>
|
||||||
|
<div class="file-icons-container">
|
||||||
|
${files.map(file => renderFileIcon(file)).join('')}
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка загрузки файлов:', error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderFileIcon(file) {
|
||||||
|
// Исправляем кодировку имени файла
|
||||||
|
const fixEncoding = (str) => {
|
||||||
|
if (!str) return '';
|
||||||
|
try {
|
||||||
|
// Пробуем разные способы декодирования
|
||||||
|
if (str.includes('Ð') || str.includes('Ñ')) {
|
||||||
|
// UTF-8 неправильно декодированный как Latin-1
|
||||||
|
return decodeURIComponent(escape(str));
|
||||||
|
}
|
||||||
|
return str;
|
||||||
|
} catch (e) {
|
||||||
|
return str;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const fileName = fixEncoding(file.original_name);
|
||||||
|
const fileSize = (file.file_size / 1024 / 1024).toFixed(2);
|
||||||
|
const uploadedBy = file.user_name;
|
||||||
|
|
||||||
|
let iconColor = '';
|
||||||
|
let iconText = '';
|
||||||
|
let textClass = '';
|
||||||
|
|
||||||
|
// Определяем расширение файла
|
||||||
|
const extension = fileName.includes('.') ?
|
||||||
|
fileName.split('.').pop().toLowerCase() :
|
||||||
|
'';
|
||||||
|
|
||||||
|
// Определяем тип файла на основе расширения
|
||||||
|
if (extension) {
|
||||||
|
switch (extension) {
|
||||||
|
case 'pdf':
|
||||||
|
iconColor = '#e74c3c';
|
||||||
|
iconText = 'PDF';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'doc':
|
||||||
|
iconColor = '#3498db';
|
||||||
|
iconText = 'DOC';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'docx':
|
||||||
|
iconColor = '#3498db';
|
||||||
|
iconText = 'DOCX';
|
||||||
|
textClass = 'medium';
|
||||||
|
break;
|
||||||
|
case 'xls':
|
||||||
|
iconColor = '#2ecc71';
|
||||||
|
iconText = 'XLS';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'xlsx':
|
||||||
|
iconColor = '#2ecc71';
|
||||||
|
iconText = 'XLSX';
|
||||||
|
textClass = 'medium';
|
||||||
|
break;
|
||||||
|
case 'csv':
|
||||||
|
iconColor = '#2ecc71';
|
||||||
|
iconText = 'CSV';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'ppt':
|
||||||
|
iconColor = '#e67e22';
|
||||||
|
iconText = 'PPT';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'pptx':
|
||||||
|
iconColor = '#e67e22';
|
||||||
|
iconText = 'PPTX';
|
||||||
|
textClass = 'medium';
|
||||||
|
break;
|
||||||
|
case 'zip':
|
||||||
|
iconColor = '#f39c12';
|
||||||
|
iconText = 'ZIP';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'rar':
|
||||||
|
iconColor = '#f39c12';
|
||||||
|
iconText = 'RAR';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case '7z':
|
||||||
|
iconColor = '#f39c12';
|
||||||
|
iconText = '7Z';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'tar':
|
||||||
|
iconColor = '#f39c12';
|
||||||
|
iconText = 'TAR';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'gz':
|
||||||
|
iconColor = '#f39c12';
|
||||||
|
iconText = 'GZ';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'txt':
|
||||||
|
iconColor = '#95a5a6';
|
||||||
|
iconText = 'TXT';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'log':
|
||||||
|
iconColor = '#95a5a6';
|
||||||
|
iconText = 'LOG';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'md':
|
||||||
|
iconColor = '#95a5a6';
|
||||||
|
iconText = 'MD';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'jpg':
|
||||||
|
iconColor = '#9b59b6';
|
||||||
|
iconText = 'JPG';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'jpeg':
|
||||||
|
iconColor = '#9b59b6';
|
||||||
|
iconText = 'JPEG';
|
||||||
|
textClass = 'medium';
|
||||||
|
break;
|
||||||
|
case 'png':
|
||||||
|
iconColor = '#9b59b6';
|
||||||
|
iconText = 'PNG';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'gif':
|
||||||
|
iconColor = '#9b59b6';
|
||||||
|
iconText = 'GIF';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'bmp':
|
||||||
|
iconColor = '#9b59b6';
|
||||||
|
iconText = 'BMP';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'svg':
|
||||||
|
iconColor = '#9b59b6';
|
||||||
|
iconText = 'SVG';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'webp':
|
||||||
|
iconColor = '#9b59b6';
|
||||||
|
iconText = 'WEBP';
|
||||||
|
textClass = 'medium';
|
||||||
|
break;
|
||||||
|
case 'mp3':
|
||||||
|
iconColor = '#1abc9c';
|
||||||
|
iconText = 'MP3';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'wav':
|
||||||
|
iconColor = '#1abc9c';
|
||||||
|
iconText = 'WAV';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'ogg':
|
||||||
|
iconColor = '#1abc9c';
|
||||||
|
iconText = 'OGG';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'flac':
|
||||||
|
iconColor = '#1abc9c';
|
||||||
|
iconText = 'FLAC';
|
||||||
|
textClass = 'medium';
|
||||||
|
break;
|
||||||
|
case 'mp4':
|
||||||
|
iconColor = '#d35400';
|
||||||
|
iconText = 'MP4';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'avi':
|
||||||
|
iconColor = '#d35400';
|
||||||
|
iconText = 'AVI';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'mkv':
|
||||||
|
iconColor = '#d35400';
|
||||||
|
iconText = 'MKV';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'mov':
|
||||||
|
iconColor = '#d35400';
|
||||||
|
iconText = 'MOV';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'wmv':
|
||||||
|
iconColor = '#d35400';
|
||||||
|
iconText = 'WMV';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'exe':
|
||||||
|
iconColor = '#c0392b';
|
||||||
|
iconText = 'EXE';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'msi':
|
||||||
|
iconColor = '#c0392b';
|
||||||
|
iconText = 'MSI';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'js':
|
||||||
|
iconColor = '#2980b9';
|
||||||
|
iconText = 'JS';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'html':
|
||||||
|
iconColor = '#2980b9';
|
||||||
|
iconText = 'HTML';
|
||||||
|
textClass = 'medium';
|
||||||
|
break;
|
||||||
|
case 'css':
|
||||||
|
iconColor = '#2980b9';
|
||||||
|
iconText = 'CSS';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'php':
|
||||||
|
iconColor = '#2980b9';
|
||||||
|
iconText = 'PHP';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'py':
|
||||||
|
iconColor = '#2980b9';
|
||||||
|
iconText = 'PY';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'java':
|
||||||
|
iconColor = '#2980b9';
|
||||||
|
iconText = 'JAVA';
|
||||||
|
textClass = 'medium';
|
||||||
|
break;
|
||||||
|
case 'json':
|
||||||
|
iconColor = '#8e44ad';
|
||||||
|
iconText = 'JSON';
|
||||||
|
textClass = 'medium';
|
||||||
|
break;
|
||||||
|
case 'xml':
|
||||||
|
iconColor = '#8e44ad';
|
||||||
|
iconText = 'XML';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'yml':
|
||||||
|
iconColor = '#8e44ad';
|
||||||
|
iconText = 'YML';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'yaml':
|
||||||
|
iconColor = '#8e44ad';
|
||||||
|
iconText = 'YAML';
|
||||||
|
textClass = 'medium';
|
||||||
|
break;
|
||||||
|
case 'sql':
|
||||||
|
iconColor = '#27ae60';
|
||||||
|
iconText = 'SQL';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'db':
|
||||||
|
iconColor = '#27ae60';
|
||||||
|
iconText = 'DB';
|
||||||
|
textClass = 'short';
|
||||||
|
break;
|
||||||
|
case 'sqlite':
|
||||||
|
iconColor = '#27ae60';
|
||||||
|
iconText = 'SQLITE';
|
||||||
|
textClass = 'long';
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
// Для других расширений используем расширение или первые 4 символа
|
||||||
|
iconColor = '#7f8c8d';
|
||||||
|
iconText = extension.length > 4 ?
|
||||||
|
extension.substring(0, 4).toUpperCase() :
|
||||||
|
extension.toUpperCase();
|
||||||
|
|
||||||
|
// Определяем класс по длине текста
|
||||||
|
if (iconText.length <= 2) {
|
||||||
|
textClass = 'short';
|
||||||
|
} else if (iconText.length <= 4) {
|
||||||
|
textClass = 'medium';
|
||||||
|
} else {
|
||||||
|
textClass = 'long';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Если нет расширения
|
||||||
|
iconColor = '#7f8c8d';
|
||||||
|
iconText = 'ФАЙЛ';
|
||||||
|
textClass = 'short';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Исправляем кодировку для отображения
|
||||||
|
const safeFileName = fileName;
|
||||||
|
const displayFileName = truncateFileName(safeFileName);
|
||||||
|
|
||||||
|
return `
|
||||||
|
<a href="/api/files/${file.id}/download"
|
||||||
|
download="${encodeURIComponent(safeFileName)}"
|
||||||
|
class="file-icon-container"
|
||||||
|
title="${safeFileName} (${fileSize} MB) - Загрузил: ${uploadedBy}">
|
||||||
|
<div class="file-icon" style="background: ${iconColor}">
|
||||||
|
<span class="file-extension ${textClass}">${iconText}</span>
|
||||||
|
</div>
|
||||||
|
<div class="file-name">${displayFileName}</div>
|
||||||
|
</a>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
function truncateFileName(fileName, maxLength = 20) {
|
||||||
|
if (fileName.length <= maxLength) return fileName;
|
||||||
|
const extension = fileName.split('.').pop();
|
||||||
|
const name = fileName.substring(0, fileName.lastIndexOf('.'));
|
||||||
|
const truncatedName = name.substring(0, maxLength - extension.length - 3) + '...';
|
||||||
|
return truncatedName + '.' + extension;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Вспомогательная функция для форматирования размера файла
|
||||||
|
function formatFileSize(bytes) {
|
||||||
|
if (bytes === 0) return '0 Bytes';
|
||||||
|
const k = 1024;
|
||||||
|
const sizes = ['Bytes', 'KB', 'MB', 'GB'];
|
||||||
|
const i = Math.floor(Math.log(bytes) / Math.log(k));
|
||||||
|
return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];
|
||||||
|
}
|
||||||
@@ -5,26 +5,29 @@
|
|||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
<title>School CRM - Управление задачами</title>
|
<title>School CRM - Управление задачами</title>
|
||||||
<link rel="stylesheet" href="style.css">
|
<link rel="stylesheet" href="style.css">
|
||||||
|
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css">
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<div id="login-modal" class="modal">
|
<div id="login-modal" class="modal">
|
||||||
<div class="modal-content">
|
<div class="modal-content">
|
||||||
<h2>Вход в School CRM</h2>
|
<h2><i class="fas fa-sign-in-alt"></i> Вход в School CRM</h2>
|
||||||
<form id="login-form">
|
<form id="login-form">
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label for="login">Логин:</label>
|
<label for="login"><i class="fas fa-user"></i> Логин:</label>
|
||||||
<input type="text" id="login" name="login" required>
|
<input type="text" id="login" name="login" required>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label for="password">Пароль:</label>
|
<label for="password"><i class="fas fa-lock"></i> Пароль:</label>
|
||||||
<input type="password" id="password" name="password" required>
|
<input type="password" id="password" name="password" required>
|
||||||
</div>
|
</div>
|
||||||
<button type="submit">Войти</button>
|
<button type="submit" class="btn-primary">
|
||||||
|
<i class="fas fa-sign-in-alt"></i> Войти
|
||||||
|
</button>
|
||||||
</form>
|
</form>
|
||||||
<div class="test-users">
|
<div class="test-users">
|
||||||
<h3>Управление задачами</h3>
|
<h3><i class="fas fa-users"></i> Управление задачами</h3>
|
||||||
<ul>
|
<ul>
|
||||||
<li><strong>@2025 </strong>МАОУ - СОШ № 25</li>
|
<li><strong><i class="fas fa-school"></i> @2025</strong> МАОУ - СОШ № 25</li>
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -32,32 +35,53 @@
|
|||||||
|
|
||||||
<div class="container">
|
<div class="container">
|
||||||
<header>
|
<header>
|
||||||
<h1>School CRM - Управление задачами</h1>
|
<div class="header-top">
|
||||||
|
<h1><i class="fas fa-tasks"></i> School CRM - Управление задачами</h1>
|
||||||
<div class="user-info">
|
<div class="user-info">
|
||||||
<span id="current-user"></span>
|
<span id="current-user"></span>
|
||||||
<button onclick="logout()">Выйти</button>
|
<button onclick="logout()" class="btn-logout">
|
||||||
|
<i class="fas fa-sign-out-alt"></i> Выйти
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<nav>
|
<nav>
|
||||||
<button onclick="showSection('tasks')">Задачи</button>
|
<button onclick="showSection('tasks')" class="nav-btn">
|
||||||
<button onclick="showSection('create-task')">Создать задачу</button>
|
<i class="fas fa-list"></i> Задачи
|
||||||
<button onclick="showTasksWithoutDate()" id="tasks-no-date-btn">Задачи без срока</button>
|
</button>
|
||||||
<button onclick="showKanbanSection()" class="nav-btn">📋 Канбан</button>
|
<button onclick="showSection('create-task')" class="nav-btn">
|
||||||
<button onclick="showSection('logs')">Лог активности</button>
|
<i class="fas fa-plus-circle"></i> Создать задачу
|
||||||
<button onclick="window.location.href = '/admin'" style="background: linear-gradient(135deg, #e74c3c, #c0392b);">Админ-панель</button>
|
</button>
|
||||||
|
<button onclick="showTasksWithoutDate()" class="nav-btn" id="tasks-no-date-btn">
|
||||||
|
<i class="fas fa-clock"></i> Задачи без срока
|
||||||
|
</button>
|
||||||
|
<button onclick="showKanbanSection()" class="nav-btn">
|
||||||
|
<i class="fas fa-columns"></i> Канбан
|
||||||
|
</button>
|
||||||
|
<button onclick="showSection('profile')" class="nav-btn" id="profile-btn">
|
||||||
|
<i class="fas fa-user-circle"></i> Личный кабинет
|
||||||
|
</button>
|
||||||
|
<!--
|
||||||
|
<button onclick="showSection('logs')" class="nav-btn">
|
||||||
|
<i class="fas fa-history"></i> Лог активности
|
||||||
|
</button>
|
||||||
|
<button onclick="window.location.href = '/admin'" class="nav-btn btn-admin">
|
||||||
|
<i class="fas fa-cog"></i> Админ-панель
|
||||||
|
</button>
|
||||||
|
-->
|
||||||
</nav>
|
</nav>
|
||||||
</header>
|
</header>
|
||||||
|
|
||||||
<main>
|
<main>
|
||||||
<section id="tasks-section" class="section">
|
<section id="tasks-section" class="section">
|
||||||
<h2>Все задачи</h2>
|
<h2><i class="fas fa-tasks"></i> Все задачи</h2>
|
||||||
<div id="tasks-controls">
|
<div id="tasks-controls">
|
||||||
<div class="filters">
|
<div class="filters">
|
||||||
<div class="filter-group">
|
<div class="filter-group">
|
||||||
<label for="search-tasks">Поиск:</label>
|
<label for="search-tasks"><i class="fas fa-search"></i> Поиск:</label>
|
||||||
<input type="text" id="search-tasks" placeholder="Поиск по названию и описанию..." oninput="loadTasks()">
|
<input type="text" id="search-tasks" placeholder="Поиск по названию и описанию..." oninput="loadTasks()">
|
||||||
</div>
|
</div>
|
||||||
<div class="filter-group">
|
<div class="filter-group">
|
||||||
<label for="status-filter">Статус:</label>
|
<label for="status-filter"><i class="fas fa-filter"></i> Статус:</label>
|
||||||
<select id="status-filter" onchange="loadTasks()">
|
<select id="status-filter" onchange="loadTasks()">
|
||||||
<option value="active,in_progress,assigned,overdue,rework">Все активные</option>
|
<option value="active,in_progress,assigned,overdue,rework">Все активные</option>
|
||||||
<option value="all">Все статусы</option>
|
<option value="all">Все статусы</option>
|
||||||
@@ -70,19 +94,19 @@
|
|||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
<div class="filter-group">
|
<div class="filter-group">
|
||||||
<label for="creator-filter">Заказчик:</label>
|
<label for="creator-filter"><i class="fas fa-user-tie"></i> Заказчик:</label>
|
||||||
<select id="creator-filter" onchange="loadTasks()">
|
<select id="creator-filter" onchange="loadTasks()">
|
||||||
<option value="">Все заказчики</option>
|
<option value="">Все заказчики</option>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
<div class="filter-group">
|
<div class="filter-group">
|
||||||
<label for="assignee-filter">Исполнитель:</label>
|
<label for="assignee-filter"><i class="fas fa-user-check"></i> Исполнитель:</label>
|
||||||
<select id="assignee-filter" onchange="loadTasks()">
|
<select id="assignee-filter" onchange="loadTasks()">
|
||||||
<option value="">Все исполнители</option>
|
<option value="">Все исполнители</option>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
<div class="filter-group">
|
<div class="filter-group">
|
||||||
<label for="deadline-filter">Срок выполнения:</label>
|
<label for="deadline-filter"><i class="fas fa-calendar-times"></i> Срок выполнения:</label>
|
||||||
<select id="deadline-filter" onchange="loadTasks()">
|
<select id="deadline-filter" onchange="loadTasks()">
|
||||||
<option value="">Все сроки</option>
|
<option value="">Все сроки</option>
|
||||||
<option value="48h">Менее 48 часов</option>
|
<option value="48h">Менее 48 часов</option>
|
||||||
@@ -92,59 +116,122 @@
|
|||||||
</div>
|
</div>
|
||||||
<label class="show-deleted-label" style="display: none;">
|
<label class="show-deleted-label" style="display: none;">
|
||||||
<input type="checkbox" id="show-deleted" onchange="loadTasks()">
|
<input type="checkbox" id="show-deleted" onchange="loadTasks()">
|
||||||
Показать удаленные задачи
|
<i class="fas fa-trash"></i> Показать удаленные задачи
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
<div id="tasks-list"></div>
|
<div id="tasks-list"></div>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section id="create-task-section" class="section">
|
<section id="create-task-section" class="section">
|
||||||
<h2>Создать новую задачу</h2>
|
<h2><i class="fas fa-plus-circle"></i> Создать новую задачу</h2>
|
||||||
<form id="create-task-form" enctype="multipart/form-data">
|
<form id="create-task-form" enctype="multipart/form-data">
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label for="title">Название задачи:</label>
|
<label for="title"><i class="fas fa-heading"></i> Название задачи:</label>
|
||||||
<input type="text" id="title" name="title" required>
|
<input type="text" id="title" name="title" required>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label for="description">Описание:</label>
|
<label for="description"><i class="fas fa-align-left"></i> Описание:</label>
|
||||||
<textarea id="description" name="description" rows="4"></textarea>
|
<textarea id="description" name="description" rows="4"></textarea>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label for="due-date">Дата и время выполнения:</label>
|
<label for="due-date"><i class="fas fa-calendar-alt"></i> Дата и время выполнения:</label>
|
||||||
<input type="datetime-local" id="due-date" name="dueDate" required>
|
<input type="datetime-local" id="due-date" name="dueDate" required>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label>Исполнители:</label>
|
<label><i class="fas fa-users"></i> Исполнители:</label>
|
||||||
<div class="user-search">
|
<div class="user-search">
|
||||||
<input type="text" id="user-search" placeholder="Поиск исполнителей..." oninput="filterUsers()">
|
<input type="text" id="user-search" placeholder="Поиск исполнителей..." oninput="filterUsers()">
|
||||||
|
<i class="fas fa-search"></i>
|
||||||
</div>
|
</div>
|
||||||
<div id="users-checklist" class="checkbox-group"></div>
|
<div id="users-checklist" class="checkbox-group"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label for="files">Прикрепить файлы (до 15 файлов, максимум 300MB):</label>
|
<label for="files"><i class="fas fa-paperclip"></i> Прикрепить файлы (до 15 файлов, максимум 300MB):</label>
|
||||||
|
<div class="file-upload">
|
||||||
<input type="file" id="files" name="files" multiple>
|
<input type="file" id="files" name="files" multiple>
|
||||||
|
<label for="files" class="file-upload-label">
|
||||||
|
<i class="fas fa-cloud-upload-alt"></i> Выберите файлы
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
<div id="file-list"></div>
|
<div id="file-list"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<button type="submit">Создать задачу</button>
|
<button type="submit" class="btn-primary">
|
||||||
|
<i class="fas fa-check-circle"></i> Создать задачу
|
||||||
|
</button>
|
||||||
</form>
|
</form>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section id="logs-section" class="section">
|
<section id="logs-section" class="section">
|
||||||
<h2>Лог активности</h2>
|
<h2><i class="fas fa-history"></i> Лог активности</h2>
|
||||||
<div id="logs-list"></div>
|
<div id="logs-list"></div>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
|
<section id="profile-section" class="section">
|
||||||
|
<h2><i class="fas fa-user-circle"></i> Личный кабинет</h2>
|
||||||
|
|
||||||
|
<div class="notification-settings">
|
||||||
|
<h3><i class="fas fa-bell"></i> Настройки уведомлений</h3>
|
||||||
|
<form id="notification-settings-form">
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="checkbox-label">
|
||||||
|
<input type="checkbox" id="email-notifications" name="email_notifications">
|
||||||
|
<span><i class="fas fa-envelope"></i> Email уведомления</span>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="notification-email"><i class="fas fa-at"></i> Email для уведомлений:</label>
|
||||||
|
<div class="input-with-icon">
|
||||||
|
<i class="fas fa-envelope"></i>
|
||||||
|
<input type="email" id="notification-email" name="notification_email"
|
||||||
|
placeholder="Введите email для уведомлений">
|
||||||
|
</div>
|
||||||
|
<small>Если не указано, будет использован email из профиля</small>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="checkbox-label">
|
||||||
|
<input type="checkbox" id="telegram-notifications" name="telegram_notifications" disabled>
|
||||||
|
<span><i class="fab fa-telegram"></i> Telegram уведомления (скоро)</span>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="checkbox-label">
|
||||||
|
<input type="checkbox" id="vk-notifications" name="vk_notifications" disabled>
|
||||||
|
<span><i class="fab fa-vk"></i> ВКонтакте уведомления (скоро)</span>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
<div class="form-group"><label class="checkbox-label">
|
||||||
|
<input type="checkbox" id="sberbank-notifications" name="sberbank_notifications" disabled>
|
||||||
|
<span><i class="fas fa-university"></i> Сбербанк Онлайн уведомления (скоро)</span>
|
||||||
|
</label></div>
|
||||||
|
<div class="form-group"><label class="checkbox-label">
|
||||||
|
<input type="checkbox" id="yandex-notifications" name="yandex_notifications" disabled>
|
||||||
|
<span><i class="fab fa-yandex"></i> Яндекс уведомления (скоро)</span>
|
||||||
|
</label></div>
|
||||||
|
<div class="form-group"><label class="checkbox-label">
|
||||||
|
<input type="checkbox" id="gosuslugi-notifications" name="gosuslugi_notifications" disabled>
|
||||||
|
<span><i class="fas fa-passport"></i> Госуслуги уведомления (скоро)</span>
|
||||||
|
</label></div>
|
||||||
|
<button type="submit" class="btn-primary">
|
||||||
|
<i class="fas fa-save"></i> Сохранить настройки
|
||||||
|
</button>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
</main>
|
</main>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<!-- Модальные окна остаются без изменений -->
|
||||||
<div id="edit-task-modal" class="modal">
|
<div id="edit-task-modal" class="modal">
|
||||||
<div class="modal-content">
|
<div class="modal-content">
|
||||||
<span class="close" onclick="closeEditModal()">×</span>
|
<span class="close" onclick="closeEditModal()">×</span>
|
||||||
<h3>Редактировать задачу</h3>
|
<h3><i class="fas fa-edit"></i> Редактировать задачу</h3>
|
||||||
<form id="edit-task-form" enctype="multipart/form-data">
|
<form id="edit-task-form" enctype="multipart/form-data">
|
||||||
<input type="hidden" id="edit-task-id">
|
<input type="hidden" id="edit-task-id">
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
@@ -176,7 +263,9 @@
|
|||||||
<div id="edit-file-list"></div>
|
<div id="edit-file-list"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<button type="submit">Сохранить изменения</button>
|
<button type="submit" class="btn-primary">
|
||||||
|
<i class="fas fa-save"></i> Сохранить изменения
|
||||||
|
</button>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -184,7 +273,7 @@
|
|||||||
<div id="copy-task-modal" class="modal">
|
<div id="copy-task-modal" class="modal">
|
||||||
<div class="modal-content">
|
<div class="modal-content">
|
||||||
<span class="close" onclick="closeCopyModal()">×</span>
|
<span class="close" onclick="closeCopyModal()">×</span>
|
||||||
<h3>Создать копию задачи</h3>
|
<h3><i class="fas fa-copy"></i> Создать копию задачи</h3>
|
||||||
<form id="copy-task-form">
|
<form id="copy-task-form">
|
||||||
<input type="hidden" id="copy-task-id">
|
<input type="hidden" id="copy-task-id">
|
||||||
|
|
||||||
@@ -200,7 +289,9 @@
|
|||||||
</div>
|
</div>
|
||||||
<div id="copy-users-checklist" class="checkbox-group"></div>
|
<div id="copy-users-checklist" class="checkbox-group"></div>
|
||||||
</div>
|
</div>
|
||||||
<button type="submit">Создать копию</button>
|
<button type="submit" class="btn-primary">
|
||||||
|
<i class="fas fa-copy"></i> Создать копию
|
||||||
|
</button>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -208,7 +299,7 @@
|
|||||||
<div id="edit-assignment-modal" class="modal">
|
<div id="edit-assignment-modal" class="modal">
|
||||||
<div class="modal-content">
|
<div class="modal-content">
|
||||||
<span class="close" onclick="closeEditAssignmentModal()">×</span>
|
<span class="close" onclick="closeEditAssignmentModal()">×</span>
|
||||||
<h3>Редактировать сроки исполнителя</h3>
|
<h3><i class="fas fa-clock"></i> Редактировать сроки исполнителя</h3>
|
||||||
<form id="edit-assignment-form">
|
<form id="edit-assignment-form">
|
||||||
<input type="hidden" id="edit-assignment-task-id">
|
<input type="hidden" id="edit-assignment-task-id">
|
||||||
<input type="hidden" id="edit-assignment-user-id">
|
<input type="hidden" id="edit-assignment-user-id">
|
||||||
@@ -216,7 +307,9 @@
|
|||||||
<label for="edit-assignment-due-date">Дата и время выполнения:</label>
|
<label for="edit-assignment-due-date">Дата и время выполнения:</label>
|
||||||
<input type="datetime-local" id="edit-assignment-due-date" name="dueDate" required>
|
<input type="datetime-local" id="edit-assignment-due-date" name="dueDate" required>
|
||||||
</div>
|
</div>
|
||||||
<button type="submit">Сохранить сроки</button>
|
<button type="submit" class="btn-primary">
|
||||||
|
<i class="fas fa-save"></i> Сохранить сроки
|
||||||
|
</button>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -224,27 +317,53 @@
|
|||||||
<div id="rework-task-modal" class="modal">
|
<div id="rework-task-modal" class="modal">
|
||||||
<div class="modal-content">
|
<div class="modal-content">
|
||||||
<span class="close" onclick="closeReworkModal()">×</span>
|
<span class="close" onclick="closeReworkModal()">×</span>
|
||||||
<h3>Вернуть задачу на доработку</h3>
|
<h3><i class="fas fa-redo"></i> Вернуть задачу на доработку</h3>
|
||||||
<form id="rework-task-form">
|
<form id="rework-task-form">
|
||||||
<input type="hidden" id="rework-task-id">
|
<input type="hidden" id="rework-task-id">
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label for="rework-comment">Комментарий к доработке:</label>
|
<label for="rework-comment">Комментарий к доработке:</label>
|
||||||
<textarea id="rework-comment" name="comment" rows="4" placeholder="Укажите, что нужно исправить..." required></textarea>
|
<textarea id="rework-comment" name="comment" rows="4" placeholder="Укажите, что нужно исправить..." required></textarea>
|
||||||
</div>
|
</div>
|
||||||
<button type="submit">Вернуть на доработку</button>
|
<button type="submit" class="btn-warning">
|
||||||
|
<i class="fas fa-redo"></i> Вернуть на доработку
|
||||||
|
</button>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div id="kanban-section" class="section kanban-section">
|
|
||||||
|
<div id="kanban-section" class="section kanban-section">
|
||||||
<div class="section-header">
|
<div class="section-header">
|
||||||
<h2>📋 Канбан-доска</h2>
|
<h2><i class="fas fa-columns"></i> Канбан-доска</h2>
|
||||||
<p>Перетаскивайте задачи между колонками для изменения статуса</p>
|
<p>Перетаскивайте задачи между колонками для изменения статуса</p>
|
||||||
|
<div class="kanban-controls">
|
||||||
|
<div class="kanban-filters">
|
||||||
|
<select id="kanban-filter" onchange="loadKanbanBoard()">
|
||||||
|
<option value="all">Все задачи</option>
|
||||||
|
<option value="created">Мои задачи (я создал)</option>
|
||||||
|
<option value="assigned">Назначенные мне</option>
|
||||||
|
</select>
|
||||||
|
<select id="kanban-days" onchange="loadKanbanBoard()">
|
||||||
|
<option value="7">7 дней</option>
|
||||||
|
<option value="14">14 дней</option>
|
||||||
|
<option value="30">30 дней</option>
|
||||||
|
<option value="365">Все задачи</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div id="kanban-board" class="kanban-board">
|
<div id="kanban-board" class="kanban-board">
|
||||||
<div class="loading">Загрузка Канбан-доски...</div>
|
<div class="loading">Загрузка Канбан-доски...</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<script src="script.js"></script>
|
|
||||||
|
<script src="auth.js"></script>
|
||||||
|
<script src="users.js"></script>
|
||||||
|
<script src="tasks.js"></script>
|
||||||
|
<script src="kanban.js"></script>
|
||||||
|
<script src="files.js"></script>
|
||||||
|
<script src="profile.js"></script>
|
||||||
|
<script src="ui.js"></script>
|
||||||
|
<script src="main.js"></script>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
300
public/kanban.js
Normal file
300
public/kanban.js
Normal file
@@ -0,0 +1,300 @@
|
|||||||
|
// kanban.js - Канбан-доска
|
||||||
|
let kanbanTasks = [];
|
||||||
|
let kanbanDays = 14;
|
||||||
|
let currentDraggedTask = null;
|
||||||
|
|
||||||
|
function showKanbanSection() {
|
||||||
|
showSection('kanban');
|
||||||
|
loadKanbanTasks();
|
||||||
|
}
|
||||||
|
|
||||||
|
async function loadKanbanTasks() {
|
||||||
|
try {
|
||||||
|
const daysSelect = document.getElementById('kanban-days');
|
||||||
|
const filterSelect = document.getElementById('kanban-filter');
|
||||||
|
|
||||||
|
// Если есть выбор в интерфейсе - используем его, иначе - значение по умолчанию
|
||||||
|
if (daysSelect) {
|
||||||
|
kanbanDays = parseInt(daysSelect.value) || 14;
|
||||||
|
} else {
|
||||||
|
kanbanDays = 14;
|
||||||
|
}
|
||||||
|
|
||||||
|
let filter = 'all';
|
||||||
|
if (filterSelect) {
|
||||||
|
filter = filterSelect.value;
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = await fetch(`/api/kanban-tasks?days=${kanbanDays}&filter=${filter}`);
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error(`Ошибка сервера: ${response.status}`);
|
||||||
|
}
|
||||||
|
const data = await response.json();
|
||||||
|
kanbanTasks = data.tasks || [];
|
||||||
|
renderKanban(data.filter);
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка загрузки задач для Канбана:', error);
|
||||||
|
document.getElementById('kanban-board').innerHTML = `
|
||||||
|
<div class="error-message">
|
||||||
|
❌ Ошибка загрузки Канбана: ${error.message}
|
||||||
|
<button onclick="loadKanbanTasks()" class="retry-btn">Повторить</button>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderKanban(filter = 'all') {
|
||||||
|
const container = document.getElementById('kanban-board');
|
||||||
|
|
||||||
|
// Группируем задачи по статусам (убрали 'unassigned')
|
||||||
|
const columns = {
|
||||||
|
'assigned': { title: 'Назначены', tasks: [], color: '#e74c3c' },
|
||||||
|
'in_progress': { title: 'В работе', tasks: [], color: '#f39c12' },
|
||||||
|
'rework': { title: 'На доработке', tasks: [], color: '#f1c40f' },
|
||||||
|
'overdue': { title: 'Просрочены', tasks: [], color: '#c0392b' },
|
||||||
|
'completed': { title: 'Выполнены', tasks: [], color: '#2ecc71' }
|
||||||
|
};
|
||||||
|
|
||||||
|
// Распределяем задачи по колонкам
|
||||||
|
kanbanTasks.forEach(task => {
|
||||||
|
const status = task.kanbanStatus || 'assigned';
|
||||||
|
// Преобразуем 'unassigned' в 'assigned'
|
||||||
|
const actualStatus = status === 'unassigned' ? 'assigned' : status;
|
||||||
|
|
||||||
|
if (columns[actualStatus]) {
|
||||||
|
columns[actualStatus].tasks.push(task);
|
||||||
|
} else {
|
||||||
|
// Если статус не найден, добавляем в 'assigned'
|
||||||
|
columns['assigned'].tasks.push(task);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Статистика по фильтру
|
||||||
|
let filterTitle = 'Все задачи';
|
||||||
|
if (filter === 'created') filterTitle = 'Задачи, которые я поставил';
|
||||||
|
if (filter === 'assigned') filterTitle = 'Задачи, которые мне поставили';
|
||||||
|
|
||||||
|
container.innerHTML = `
|
||||||
|
<div class="kanban-controls">
|
||||||
|
<div class="kanban-filters">
|
||||||
|
<div class="kanban-period">
|
||||||
|
<label>Период просмотра:</label>
|
||||||
|
<select id="kanban-days" onchange="loadKanbanTasks()">
|
||||||
|
${[1, 2, 3, 4, 5, 6, 7, 14, 30, 62].map(days =>
|
||||||
|
`<option value="${days}" ${days === kanbanDays ? 'selected' : ''}>${days} ${getDayWord(days)}</option>`
|
||||||
|
).join('')}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
<div class="kanban-filter-type">
|
||||||
|
<label>Показать:</label>
|
||||||
|
<select id="kanban-filter" onchange="loadKanbanTasks()">
|
||||||
|
<option value="all" ${filter === 'all' ? 'selected' : ''}>Все задачи</option>
|
||||||
|
<option value="created" ${filter === 'created' ? 'selected' : ''}>Я поставил</option>
|
||||||
|
<option value="assigned" ${filter === 'assigned' ? 'selected' : ''}>Мне поставили</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
<div class="kanban-stats">
|
||||||
|
<span class="filter-title">${filterTitle}</span>
|
||||||
|
<span class="task-count">Всего задач: ${kanbanTasks.length}</span>
|
||||||
|
<button onclick="loadKanbanTasks()" class="refresh-btn">🔄 Обновить</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="kanban-columns">
|
||||||
|
${Object.entries(columns).map(([status, column]) => `
|
||||||
|
<div class="kanban-column" data-status="${status}" ${status === 'overdue' || status === 'assigned' ? 'ondragover="return false" ondrop="return false"' : ''}>
|
||||||
|
<div class="kanban-column-header" style="background: ${column.color}">
|
||||||
|
<h3>${column.title}</h3>
|
||||||
|
<span class="kanban-count">${column.tasks.length}</span>
|
||||||
|
</div>
|
||||||
|
<div class="kanban-column-body" id="kanban-column-${status}"
|
||||||
|
${status === 'overdue' || status === 'assigned' ? 'style="opacity: 0.6; cursor: not-allowed;"' : ''}>
|
||||||
|
${renderKanbanCards(column.tasks, filter)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`).join('')}
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
|
||||||
|
// Делаем колонки перетаскиваемыми (кроме 'overdue' и 'assigned')
|
||||||
|
makeKanbanDraggable();
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderKanbanCards(tasks, filter) {
|
||||||
|
if (tasks.length === 0) {
|
||||||
|
return '<div class="kanban-empty">Нет задач</div>';
|
||||||
|
}
|
||||||
|
|
||||||
|
return tasks.map(task => {
|
||||||
|
// Определяем иконку роли
|
||||||
|
let roleIcon = '';
|
||||||
|
let roleTitle = '';
|
||||||
|
|
||||||
|
if (task.userRole === 'creator') {
|
||||||
|
roleIcon = '👤';
|
||||||
|
roleTitle = 'Вы поставили эту задачу';
|
||||||
|
} else if (task.userRole === 'assignee') {
|
||||||
|
roleIcon = '🎯';
|
||||||
|
roleTitle = 'Вам поставили эту задачу';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Исправление: безопасное получение имени пользователя
|
||||||
|
const userName = task.assignments && task.assignments.length > 0 && task.assignments[0]?.user_name
|
||||||
|
? task.assignments[0].user_name
|
||||||
|
: 'Неизвестно';
|
||||||
|
|
||||||
|
// Исправление: безопасное получение первого символа имени
|
||||||
|
const userInitial = userName && userName.length > 0 ? userName.charAt(0) : '?';
|
||||||
|
|
||||||
|
return `
|
||||||
|
<div class="kanban-card" draggable="true" data-task-id="${task.id}">
|
||||||
|
<div class="kanban-card-header">
|
||||||
|
<div class="kanban-task-id">#${task.id}</div>
|
||||||
|
<div class="kanban-task-role" title="${roleTitle}">${roleIcon}</div>
|
||||||
|
<div class="kanban-task-actions">
|
||||||
|
<button onclick="openKanbanTask(${task.id})" title="Открыть">👁️</button>
|
||||||
|
<button onclick="copyKanbanTask(${task.id})" title="Копировать">📋</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="kanban-task-title" onclick="openKanbanTask(${task.id})">
|
||||||
|
${task.title || 'Без названия'}
|
||||||
|
</div>
|
||||||
|
<div class="kanban-task-info">
|
||||||
|
<div class="kanban-deadline">
|
||||||
|
${task.due_date ? `<span class="kanban-date">📅 ${formatDate(task.due_date)}</span>` : '<span class="kanban-no-date">Без срока</span>'}
|
||||||
|
</div>
|
||||||
|
<div class="kanban-assignees">
|
||||||
|
${task.assignments && task.assignments.length > 0 ?
|
||||||
|
task.assignments.slice(0, 3).map(a => {
|
||||||
|
// Исправление: безопасное получение имени исполнителя
|
||||||
|
const assigneeName = a.user_name || 'Неизвестно';
|
||||||
|
const assigneeInitial = assigneeName && assigneeName.length > 0 ? assigneeName.charAt(0) : '?';
|
||||||
|
return `<span class="kanban-assignee" title="${assigneeName}">${assigneeInitial}</span>`;
|
||||||
|
}).join('') :
|
||||||
|
'<span class="kanban-no-assignee">👤</span>'
|
||||||
|
}
|
||||||
|
${task.assignments && task.assignments.length > 3 ?
|
||||||
|
`<span class="kanban-more-assignees">+${task.assignments.length - 3}</span>` : ''
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="kanban-task-footer">
|
||||||
|
<span class="kanban-creator">👤 ${task.creator_name || 'Неизвестно'}</span>
|
||||||
|
${task.files && task.files.length > 0 ?
|
||||||
|
`<span class="kanban-files">📎 ${task.files.length}</span>` : ''
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}).join('');
|
||||||
|
}
|
||||||
|
|
||||||
|
function getDayWord(days) {
|
||||||
|
if (days === 1) return 'день';
|
||||||
|
if (days >= 2 && days <= 4) return 'дня';
|
||||||
|
return 'дней';
|
||||||
|
}
|
||||||
|
|
||||||
|
function makeKanbanDraggable() {
|
||||||
|
const cards = document.querySelectorAll('.kanban-card');
|
||||||
|
const columns = document.querySelectorAll('.kanban-column-body:not([style*="opacity: 0.6"])');
|
||||||
|
|
||||||
|
cards.forEach(card => {
|
||||||
|
card.addEventListener('dragstart', (e) => {
|
||||||
|
e.dataTransfer.setData('text/plain', card.dataset.taskId);
|
||||||
|
card.classList.add('dragging');
|
||||||
|
});
|
||||||
|
|
||||||
|
card.addEventListener('dragend', () => {
|
||||||
|
card.classList.remove('dragging');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
columns.forEach(column => {
|
||||||
|
const status = column.parentElement.dataset.status;
|
||||||
|
|
||||||
|
// Запрещаем перетаскивание в 'overdue' и 'assigned'
|
||||||
|
if (status === 'overdue' || status === 'assigned') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
column.addEventListener('dragover', (e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
const draggingCard = document.querySelector('.dragging');
|
||||||
|
if (draggingCard) {
|
||||||
|
column.appendChild(draggingCard);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
column.addEventListener('drop', async (e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
const taskId = e.dataTransfer.getData('text/plain');
|
||||||
|
const newStatus = column.parentElement.dataset.status;
|
||||||
|
|
||||||
|
if (taskId) {
|
||||||
|
try {
|
||||||
|
// Запрещаем установку статуса 'overdue' и 'assigned'
|
||||||
|
if (newStatus === 'overdue' || newStatus === 'assigned') {
|
||||||
|
alert('Невозможно изменить статус задачи на "Просрочены" или "Назначены" через Канбан');
|
||||||
|
// Возвращаем задачу в исходное положение
|
||||||
|
loadKanbanTasks();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Обновляем статус на сервере
|
||||||
|
const response = await fetch(`/api/kanban-tasks/${taskId}/status`, {
|
||||||
|
method: 'PUT',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
body: JSON.stringify({ status: newStatus })
|
||||||
|
});
|
||||||
|
|
||||||
|
if (response.ok) {
|
||||||
|
// Перезагружаем Канбан
|
||||||
|
loadKanbanTasks();
|
||||||
|
} else {
|
||||||
|
const error = await response.json();
|
||||||
|
alert(`Ошибка обновления статуса: ${error.error || 'Неизвестная ошибка'}`);
|
||||||
|
// Возвращаем задачу в исходное положение
|
||||||
|
loadKanbanTasks();
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка обновления статуса:', error);
|
||||||
|
alert('Ошибка обновления статуса');
|
||||||
|
loadKanbanTasks();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function openKanbanTask(taskId) {
|
||||||
|
// Находим задачу и открываем её в основном интерфейсе
|
||||||
|
const task = kanbanTasks.find(t => t.id == taskId);
|
||||||
|
if (task) {
|
||||||
|
showSection('tasks');
|
||||||
|
// Прокручиваем к задаче
|
||||||
|
setTimeout(() => {
|
||||||
|
const taskElement = document.querySelector(`.task-card[data-task-id="${taskId}"]`);
|
||||||
|
if (taskElement) {
|
||||||
|
taskElement.scrollIntoView({ behavior: 'smooth', block: 'center' });
|
||||||
|
// Раскрываем задачу если она свернута
|
||||||
|
if (!expandedTasks.has(taskId)) {
|
||||||
|
toggleTask(taskId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, 100);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function copyKanbanTask(taskId) {
|
||||||
|
openCopyModal(taskId);
|
||||||
|
}
|
||||||
|
|
||||||
|
function formatDate(dateString) {
|
||||||
|
if (!dateString) return '';
|
||||||
|
const date = new Date(dateString);
|
||||||
|
return date.toLocaleDateString('ru-RU');
|
||||||
|
}
|
||||||
38
public/main.js
Normal file
38
public/main.js
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
// main.js - Главный файл инициализации
|
||||||
|
document.addEventListener('DOMContentLoaded', function() {
|
||||||
|
checkAuth();
|
||||||
|
setupEventListeners();
|
||||||
|
|
||||||
|
// Устанавливаем дату по умолчанию для создания задачи (через 3 дня)
|
||||||
|
const defaultDate = new Date();
|
||||||
|
defaultDate.setDate(defaultDate.getDate() + 3);
|
||||||
|
document.getElementById('due-date').value = defaultDate.toISOString().substring(0, 16);
|
||||||
|
|
||||||
|
// По умолчанию показываем секцию задач
|
||||||
|
showSection('tasks');
|
||||||
|
});
|
||||||
|
|
||||||
|
function setupEventListeners() {
|
||||||
|
// Форма входа
|
||||||
|
document.getElementById('login-form').addEventListener('submit', login);
|
||||||
|
|
||||||
|
// Формы задач
|
||||||
|
document.getElementById('create-task-form').addEventListener('submit', createTask);
|
||||||
|
document.getElementById('edit-task-form').addEventListener('submit', updateTask);
|
||||||
|
document.getElementById('copy-task-form').addEventListener('submit', copyTask);
|
||||||
|
document.getElementById('edit-assignment-form').addEventListener('submit', updateAssignment);
|
||||||
|
document.getElementById('rework-task-form').addEventListener('submit', sendForRework);
|
||||||
|
|
||||||
|
// Файлы
|
||||||
|
document.getElementById('files').addEventListener('change', updateFileList);
|
||||||
|
document.getElementById('edit-files').addEventListener('change', updateEditFileList);
|
||||||
|
|
||||||
|
// Настройки уведомлений
|
||||||
|
const notificationForm = document.getElementById('notification-settings-form');
|
||||||
|
if (notificationForm) {
|
||||||
|
notificationForm.addEventListener('submit', saveNotificationSettings);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Инициализация загрузки файлов
|
||||||
|
initializeFileUploads();
|
||||||
|
}
|
||||||
116
public/profile.js
Normal file
116
public/profile.js
Normal file
@@ -0,0 +1,116 @@
|
|||||||
|
// profile.js - Личный кабинет и настройки
|
||||||
|
|
||||||
|
// Личный кабинет
|
||||||
|
function showProfileSection() {
|
||||||
|
showSection('profile');
|
||||||
|
loadUserProfile();
|
||||||
|
loadNotificationSettings();
|
||||||
|
}
|
||||||
|
|
||||||
|
async function loadUserProfile() {
|
||||||
|
try {
|
||||||
|
const response = await fetch('/api/user');
|
||||||
|
const data = await response.json();
|
||||||
|
|
||||||
|
if (data.user) {
|
||||||
|
const userInfo = document.getElementById('user-profile-info');
|
||||||
|
userInfo.innerHTML = `
|
||||||
|
<div class="profile-card">
|
||||||
|
<div class="profile-field">
|
||||||
|
<i class="fas fa-user"></i>
|
||||||
|
<div>
|
||||||
|
<strong>Имя:</strong>
|
||||||
|
<p>${data.user.name}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="profile-field">
|
||||||
|
<i class="fas fa-at"></i>
|
||||||
|
<div>
|
||||||
|
<strong>Логин:</strong>
|
||||||
|
<p>${data.user.login}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="profile-field">
|
||||||
|
<i class="fas fa-envelope"></i>
|
||||||
|
<div>
|
||||||
|
<strong>Email:</strong>
|
||||||
|
<p>${data.user.email || 'Не указан'}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="profile-field">
|
||||||
|
<i class="fas fa-user-tag"></i>
|
||||||
|
<div>
|
||||||
|
<strong>Роль:</strong>
|
||||||
|
<p>${data.user.role === 'admin' ? 'Администратор' : 'Учитель'}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="profile-field">
|
||||||
|
<i class="fas fa-key"></i>
|
||||||
|
<div>
|
||||||
|
<strong>Тип авторизации:</strong>
|
||||||
|
<p>${data.user.auth_type === 'ldap' ? 'LDAP' : 'Локальная'}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
${data.user.groups && data.user.groups.length > 0 ? `
|
||||||
|
<div class="profile-field">
|
||||||
|
<i class="fas fa-users"></i>
|
||||||
|
<div>
|
||||||
|
<strong>Группы:</strong>
|
||||||
|
<p>${Array.isArray(data.user.groups) ? data.user.groups.join(', ') : data.user.groups}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
` : ''}
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка загрузки профиля:', error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Настройки уведомлений
|
||||||
|
async function loadNotificationSettings() {
|
||||||
|
try {
|
||||||
|
const response = await fetch('/api/user/settings');
|
||||||
|
const settings = await response.json();
|
||||||
|
|
||||||
|
document.getElementById('email-notifications').checked = settings.email_notifications;
|
||||||
|
document.getElementById('notification-email').value = settings.notification_email || '';
|
||||||
|
document.getElementById('telegram-notifications').checked = settings.telegram_notifications;
|
||||||
|
document.getElementById('vk-notifications').checked = settings.vk_notifications;
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка загрузки настроек:', error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function saveNotificationSettings(event) {
|
||||||
|
event.preventDefault();
|
||||||
|
|
||||||
|
const settings = {
|
||||||
|
email_notifications: document.getElementById('email-notifications').checked,
|
||||||
|
notification_email: document.getElementById('notification-email').value.trim(),
|
||||||
|
telegram_notifications: document.getElementById('telegram-notifications').checked,
|
||||||
|
vk_notifications: document.getElementById('vk-notifications').checked
|
||||||
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch('/api/user/settings', {
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json'
|
||||||
|
},
|
||||||
|
body: JSON.stringify(settings)
|
||||||
|
});
|
||||||
|
|
||||||
|
const result = await response.json();
|
||||||
|
|
||||||
|
if (result.success) {
|
||||||
|
alert('Настройки уведомлений сохранены!');
|
||||||
|
} else {
|
||||||
|
alert('Ошибка сохранения настроек: ' + (result.error || 'Неизвестная ошибка'));
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка сохранения настроек:', error);
|
||||||
|
alert('Ошибка сохранения настроек');
|
||||||
|
}
|
||||||
|
}
|
||||||
1932
public/script.js
1932
public/script.js
File diff suppressed because it is too large
Load Diff
@@ -2212,3 +2212,68 @@ button.reopen-btn:hover {
|
|||||||
transform: none !important;
|
transform: none !important;
|
||||||
box-shadow: 0 2px 4px rgba(0,0,0,0.1) !important;
|
box-shadow: 0 2px 4px rgba(0,0,0,0.1) !important;
|
||||||
}
|
}
|
||||||
|
/* Добавим в существующий style.css */
|
||||||
|
|
||||||
|
.profile-card {
|
||||||
|
background: white;
|
||||||
|
border-radius: 10px;
|
||||||
|
padding: 20px;
|
||||||
|
box-shadow: 0 2px 10px rgba(0,0,0,0.1);
|
||||||
|
margin-bottom: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.profile-card p {
|
||||||
|
margin: 10px 0;
|
||||||
|
padding: 5px 0;
|
||||||
|
border-bottom: 1px solid #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
.profile-card p:last-child {
|
||||||
|
border-bottom: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.notification-settings {
|
||||||
|
background: white;
|
||||||
|
border-radius: 10px;
|
||||||
|
padding: 20px;
|
||||||
|
box-shadow: 0 2px 10px rgba(0,0,0,0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.checkbox-label {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
margin: 10px 0;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.checkbox-label input[type="checkbox"] {
|
||||||
|
margin-right: 10px;
|
||||||
|
width: 18px;
|
||||||
|
height: 18px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.checkbox-label span {
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
#notification-email {
|
||||||
|
width: 100%;
|
||||||
|
padding: 8px 12px;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
border-radius: 4px;
|
||||||
|
font-size: 14px;
|
||||||
|
margin-top: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#notification-email:focus {
|
||||||
|
outline: none;
|
||||||
|
border-color: #667eea;
|
||||||
|
box-shadow: 0 0 0 2px rgba(102, 126, 234, 0.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
small {
|
||||||
|
color: #666;
|
||||||
|
font-size: 12px;
|
||||||
|
display: block;
|
||||||
|
margin-top: 5px;
|
||||||
|
}
|
||||||
558
public/tasks.js
Normal file
558
public/tasks.js
Normal file
@@ -0,0 +1,558 @@
|
|||||||
|
// tasks.js - Основные операции с задачами
|
||||||
|
let tasks = [];
|
||||||
|
let expandedTasks = new Set();
|
||||||
|
let showingTasksWithoutDate = false;
|
||||||
|
|
||||||
|
async function loadTasks() {
|
||||||
|
try {
|
||||||
|
showingTasksWithoutDate = false;
|
||||||
|
const btn = document.getElementById('tasks-no-date-btn');
|
||||||
|
if (btn) btn.classList.remove('active');
|
||||||
|
|
||||||
|
const search = document.getElementById('search-tasks')?.value || '';
|
||||||
|
const statusFilter = document.getElementById('status-filter')?.value || 'active,in_progress,assigned,overdue,rework';
|
||||||
|
const creatorFilter = document.getElementById('creator-filter')?.value || '';
|
||||||
|
const assigneeFilter = document.getElementById('assignee-filter')?.value || '';
|
||||||
|
const deadlineFilter = document.getElementById('deadline-filter')?.value || '';
|
||||||
|
const showDeleted = document.getElementById('show-deleted')?.checked || false;
|
||||||
|
|
||||||
|
let url = '/api/tasks?';
|
||||||
|
if (search) url += `search=${encodeURIComponent(search)}&`;
|
||||||
|
if (statusFilter) url += `status=${encodeURIComponent(statusFilter)}&`;
|
||||||
|
if (creatorFilter) url += `creator=${encodeURIComponent(creatorFilter)}&`;
|
||||||
|
if (assigneeFilter) url += `assignee=${encodeURIComponent(assigneeFilter)}&`;
|
||||||
|
if (deadlineFilter) url += `deadline=${encodeURIComponent(deadlineFilter)}&`;
|
||||||
|
if (showDeleted) url += `showDeleted=true&`;
|
||||||
|
|
||||||
|
const response = await fetch(url);
|
||||||
|
tasks = await response.json();
|
||||||
|
|
||||||
|
// Загружаем файлы для всех задач
|
||||||
|
await Promise.all(tasks.map(async (task) => {
|
||||||
|
try {
|
||||||
|
const filesResponse = await fetch(`/api/tasks/${task.id}/files`);
|
||||||
|
if (filesResponse.ok) {
|
||||||
|
task.files = await filesResponse.json();
|
||||||
|
} else {
|
||||||
|
task.files = [];
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error(`Ошибка загрузки файлов для задачи ${task.id}:`, error);
|
||||||
|
task.files = [];
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
|
||||||
|
renderTasks();
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка загрузки задач:', error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function showTasksWithoutDate() {
|
||||||
|
showingTasksWithoutDate = true;
|
||||||
|
const btn = document.getElementById('tasks-no-date-btn');
|
||||||
|
if (btn) btn.classList.add('active');
|
||||||
|
loadTasksWithoutDate();
|
||||||
|
}
|
||||||
|
|
||||||
|
async function loadTasksWithoutDate() {
|
||||||
|
try {
|
||||||
|
const response = await fetch('/api/tasks');
|
||||||
|
if (!response.ok) throw new Error('Ошибка загрузки задач');
|
||||||
|
|
||||||
|
const allTasks = await response.json();
|
||||||
|
tasks = allTasks.filter(task => {
|
||||||
|
const hasTaskDueDate = !task.due_date;
|
||||||
|
const hasAssignmentDueDates = task.assignments &&
|
||||||
|
task.assignments.every(assignment => !assignment.due_date);
|
||||||
|
return hasTaskDueDate && hasAssignmentDueDates;
|
||||||
|
});
|
||||||
|
|
||||||
|
// Загружаем файлы для всех задач
|
||||||
|
await Promise.all(tasks.map(async (task) => {
|
||||||
|
try {
|
||||||
|
const filesResponse = await fetch(`/api/tasks/${task.id}/files`);
|
||||||
|
if (filesResponse.ok) {
|
||||||
|
task.files = await filesResponse.json();
|
||||||
|
} else {
|
||||||
|
task.files = [];
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error(`Ошибка загрузки файлов для задачи ${task.id}:`, error);
|
||||||
|
task.files = [];
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
|
||||||
|
renderTasks();
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка загрузки задач без срока:', error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function createTask(event) {
|
||||||
|
event.preventDefault();
|
||||||
|
|
||||||
|
if (!currentUser) {
|
||||||
|
alert('Требуется аутентификация');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const formData = new FormData();
|
||||||
|
formData.append('title', document.getElementById('title').value);
|
||||||
|
formData.append('description', document.getElementById('description').value);
|
||||||
|
|
||||||
|
const dueDate = document.getElementById('due-date').value;
|
||||||
|
if (!dueDate) {
|
||||||
|
alert('Дата и время выполнения обязательны');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
formData.append('dueDate', dueDate);
|
||||||
|
|
||||||
|
// Используем selectedUsers вместо прямого доступа к DOM
|
||||||
|
if (selectedUsers.length === 0) {
|
||||||
|
alert('Выберите хотя бы одного исполнителя');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
selectedUsers.forEach(userId => {
|
||||||
|
formData.append('assignedUsers', userId);
|
||||||
|
});
|
||||||
|
|
||||||
|
const files = document.getElementById('files').files;
|
||||||
|
for (let i = 0; i < files.length; i++) {
|
||||||
|
formData.append('files', files[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch('/api/tasks', {
|
||||||
|
method: 'POST',
|
||||||
|
body: formData
|
||||||
|
});
|
||||||
|
|
||||||
|
if (response.ok) {
|
||||||
|
alert('Задача успешно создана!');
|
||||||
|
document.getElementById('create-task-form').reset();
|
||||||
|
document.getElementById('file-list').innerHTML = '';
|
||||||
|
document.getElementById('user-search').value = '';
|
||||||
|
selectedUsers = [];
|
||||||
|
renderUsersChecklist();
|
||||||
|
loadTasks();
|
||||||
|
loadActivityLogs();
|
||||||
|
showSection('tasks');
|
||||||
|
} else {
|
||||||
|
const error = await response.json();
|
||||||
|
alert(error.error || 'Ошибка создания задачи');
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка:', error);
|
||||||
|
alert('Ошибка создания задачи');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function openEditModal(taskId) {
|
||||||
|
try {
|
||||||
|
const response = await fetch(`/api/tasks/${taskId}`);
|
||||||
|
if (!response.ok) {
|
||||||
|
if (response.status === 404) {
|
||||||
|
alert('Задача не найдена или у вас нет прав доступа');
|
||||||
|
}
|
||||||
|
throw new Error('Ошибка загрузки задачи');
|
||||||
|
}
|
||||||
|
|
||||||
|
const task = await response.json();
|
||||||
|
|
||||||
|
if (!canUserEditTask(task)) {
|
||||||
|
alert('У вас нет прав для редактирования этой задачи');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
document.getElementById('edit-task-id').value = task.id;
|
||||||
|
document.getElementById('edit-title').value = task.title;
|
||||||
|
document.getElementById('edit-description').value = task.description || '';
|
||||||
|
|
||||||
|
document.getElementById('edit-due-date').value = task.due_date ? formatDateTimeForInput(task.due_date) : '';
|
||||||
|
|
||||||
|
// Устанавливаем выбранных пользователей
|
||||||
|
editSelectedUsers = task.assignments ? task.assignments.map(a => a.user_id) : [];
|
||||||
|
renderEditUsersChecklist(users);
|
||||||
|
|
||||||
|
// Показываем существующие файлы
|
||||||
|
currentEditTaskFiles = task.files || [];
|
||||||
|
updateEditFileList();
|
||||||
|
|
||||||
|
document.getElementById('edit-task-modal').style.display = 'block';
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка:', error);
|
||||||
|
alert('Ошибка загрузки задачи');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function closeEditModal() {
|
||||||
|
document.getElementById('edit-task-modal').style.display = 'none';
|
||||||
|
document.getElementById('edit-file-list').innerHTML = '';
|
||||||
|
document.getElementById('edit-user-search').value = '';
|
||||||
|
editSelectedUsers = [];
|
||||||
|
currentEditTaskFiles = [];
|
||||||
|
filterEditUsers();
|
||||||
|
}
|
||||||
|
|
||||||
|
async function updateTask(event) {
|
||||||
|
event.preventDefault();
|
||||||
|
|
||||||
|
const taskId = document.getElementById('edit-task-id').value;
|
||||||
|
const title = document.getElementById('edit-title').value;
|
||||||
|
const description = document.getElementById('edit-description').value;
|
||||||
|
const dueDate = document.getElementById('edit-due-date').value;
|
||||||
|
|
||||||
|
if (!dueDate) {
|
||||||
|
alert('Дата и время выполнения обязательны');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Используем editSelectedUsers
|
||||||
|
const assignedUserIds = editSelectedUsers;
|
||||||
|
|
||||||
|
const formData = new FormData();
|
||||||
|
formData.append('title', title);
|
||||||
|
formData.append('description', description);
|
||||||
|
formData.append('assignedUsers', JSON.stringify(assignedUserIds));
|
||||||
|
formData.append('dueDate', dueDate);
|
||||||
|
|
||||||
|
const files = document.getElementById('edit-files').files;
|
||||||
|
for (let i = 0; i < files.length; i++) {
|
||||||
|
formData.append('files', files[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch(`/api/tasks/${taskId}`, {
|
||||||
|
method: 'PUT',
|
||||||
|
body: formData
|
||||||
|
});
|
||||||
|
|
||||||
|
if (response.ok) {
|
||||||
|
alert('Задача успешно обновлена!');
|
||||||
|
closeEditModal();
|
||||||
|
loadTasks();
|
||||||
|
loadActivityLogs();
|
||||||
|
} else {
|
||||||
|
const error = await response.json();
|
||||||
|
alert(error.error || 'Ошибка обновления задачи');
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка:', error);
|
||||||
|
alert('Ошибка обновления задачи');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function openCopyModal(taskId) {
|
||||||
|
document.getElementById('copy-task-id').value = taskId;
|
||||||
|
|
||||||
|
// Устанавливаем дату по умолчанию (через 7 дней)
|
||||||
|
const defaultDate = new Date();
|
||||||
|
defaultDate.setDate(defaultDate.getDate() + 7);
|
||||||
|
document.getElementById('copy-due-date').value = defaultDate.toISOString().substring(0, 16);
|
||||||
|
|
||||||
|
// Сбрасываем выбранных пользователей
|
||||||
|
copySelectedUsers = [];
|
||||||
|
renderCopyUsersChecklist(users);
|
||||||
|
|
||||||
|
document.getElementById('copy-task-modal').style.display = 'block';
|
||||||
|
}
|
||||||
|
|
||||||
|
function closeCopyModal() {
|
||||||
|
document.getElementById('copy-task-modal').style.display = 'none';
|
||||||
|
document.getElementById('copy-user-search').value = '';
|
||||||
|
copySelectedUsers = [];
|
||||||
|
filterCopyUsers();
|
||||||
|
}
|
||||||
|
|
||||||
|
async function copyTask(event) {
|
||||||
|
event.preventDefault();
|
||||||
|
|
||||||
|
const taskId = document.getElementById('copy-task-id').value;
|
||||||
|
const dueDate = document.getElementById('copy-due-date').value;
|
||||||
|
|
||||||
|
if (!dueDate) {
|
||||||
|
alert('Дата и время выполнения обязательны для копии задачи');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Используем copySelectedUsers
|
||||||
|
const assignedUserIds = copySelectedUsers;
|
||||||
|
|
||||||
|
if (assignedUserIds.length === 0) {
|
||||||
|
alert('Выберите хотя бы одного исполнителя для копии задачи');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch(`/api/tasks/${taskId}/copy`, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
body: JSON.stringify({
|
||||||
|
assignedUsers: assignedUserIds,
|
||||||
|
dueDate: dueDate
|
||||||
|
})
|
||||||
|
});
|
||||||
|
|
||||||
|
if (response.ok) {
|
||||||
|
alert('Копия задачи успешно создана!');
|
||||||
|
closeCopyModal();
|
||||||
|
loadTasks();
|
||||||
|
loadActivityLogs();
|
||||||
|
} else {
|
||||||
|
const error = await response.json();
|
||||||
|
alert(error.error || 'Ошибка создания копии задачи');
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка:', error);
|
||||||
|
alert('Ошибка создания копии задачи');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function closeTask(taskId) {
|
||||||
|
if (!confirm('Вы уверены, что хотите закрыть эту задачу? Исполнители больше не будут видеть её.')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch(`/api/tasks/${taskId}/close`, {
|
||||||
|
method: 'POST'
|
||||||
|
});
|
||||||
|
|
||||||
|
if (response.ok) {
|
||||||
|
alert('Задача закрыта!');
|
||||||
|
loadTasks();
|
||||||
|
loadActivityLogs();
|
||||||
|
} else {
|
||||||
|
const error = await response.json();
|
||||||
|
alert(error.error || 'Ошибка закрытия задачи');
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка:', error);
|
||||||
|
alert('Ошибка закрытия задачи');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function reopenTask(taskId) {
|
||||||
|
try {
|
||||||
|
const response = await fetch(`/api/tasks/${taskId}/reopen`, {
|
||||||
|
method: 'POST'
|
||||||
|
});
|
||||||
|
|
||||||
|
if (response.ok) {
|
||||||
|
alert('Задача открыта!');
|
||||||
|
loadTasks();
|
||||||
|
loadActivityLogs();
|
||||||
|
} else {
|
||||||
|
const error = await response.json();
|
||||||
|
alert(error.error || 'Ошибка открытия задачи');
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка:', error);
|
||||||
|
alert('Ошибка открытия задачи');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function deleteTask(taskId) {
|
||||||
|
if (!confirm('Вы уверены, что хотите удалить эту задачу?')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch(`/api/tasks/${taskId}`, {
|
||||||
|
method: 'DELETE'
|
||||||
|
});
|
||||||
|
|
||||||
|
if (response.ok) {
|
||||||
|
alert('Задача удалена!');
|
||||||
|
loadTasks();
|
||||||
|
loadActivityLogs();
|
||||||
|
} else {
|
||||||
|
const error = await response.json();
|
||||||
|
alert(error.error || 'Ошибка удаления задачи');
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка:', error);
|
||||||
|
alert('Ошибка удаления задачи');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function restoreTask(taskId) {
|
||||||
|
try {
|
||||||
|
const response = await fetch(`/api/tasks/${taskId}/restore`, {
|
||||||
|
method: 'POST'
|
||||||
|
});
|
||||||
|
|
||||||
|
if (response.ok) {
|
||||||
|
alert('Задача восстановлена!');
|
||||||
|
loadTasks();
|
||||||
|
loadActivityLogs();
|
||||||
|
} else {
|
||||||
|
const error = await response.json();
|
||||||
|
alert(error.error || 'Ошибка восстановления задачи');
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка:', error);
|
||||||
|
alert('Ошибка восстановления задачи');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function openEditAssignmentModal(taskId, userId) {
|
||||||
|
const task = tasks.find(t => t.id === taskId);
|
||||||
|
if (!task) return;
|
||||||
|
|
||||||
|
const assignment = task.assignments.find(a => a.user_id === userId);
|
||||||
|
if (!assignment) return;
|
||||||
|
|
||||||
|
document.getElementById('edit-assignment-task-id').value = taskId;
|
||||||
|
document.getElementById('edit-assignment-user-id').value = userId;
|
||||||
|
document.getElementById('edit-assignment-due-date').value = assignment.due_date ? formatDateTimeForInput(assignment.due_date) : '';
|
||||||
|
|
||||||
|
document.getElementById('edit-assignment-modal').style.display = 'block';
|
||||||
|
}
|
||||||
|
|
||||||
|
function closeEditAssignmentModal() {
|
||||||
|
document.getElementById('edit-assignment-modal').style.display = 'none';
|
||||||
|
}
|
||||||
|
|
||||||
|
async function updateAssignment(event) {
|
||||||
|
event.preventDefault();
|
||||||
|
|
||||||
|
const taskId = document.getElementById('edit-assignment-task-id').value;
|
||||||
|
const userId = document.getElementById('edit-assignment-user-id').value;
|
||||||
|
const dueDate = document.getElementById('edit-assignment-due-date').value;
|
||||||
|
|
||||||
|
if (!dueDate) {
|
||||||
|
alert('Дата и время выполнения обязательны');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch(`/api/tasks/${taskId}/assignment/${userId}`, {
|
||||||
|
method: 'PUT',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
body: JSON.stringify({
|
||||||
|
dueDate: dueDate
|
||||||
|
})
|
||||||
|
});
|
||||||
|
|
||||||
|
if (response.ok) {
|
||||||
|
alert('Сроки исполнителя обновлены!');
|
||||||
|
closeEditAssignmentModal();
|
||||||
|
loadTasks();
|
||||||
|
loadActivityLogs();
|
||||||
|
} else {
|
||||||
|
const error = await response.json();
|
||||||
|
alert(error.error || 'Ошибка обновления сроков');
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка:', error);
|
||||||
|
alert('Ошибка обновления сроков');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function openReworkModal(taskId) {
|
||||||
|
document.getElementById('rework-task-id').value = taskId;
|
||||||
|
document.getElementById('rework-task-modal').style.display = 'block';
|
||||||
|
}
|
||||||
|
|
||||||
|
function closeReworkModal() {
|
||||||
|
document.getElementById('rework-task-modal').style.display = 'none';
|
||||||
|
document.getElementById('rework-comment').value = '';
|
||||||
|
}
|
||||||
|
|
||||||
|
async function sendForRework(event) {
|
||||||
|
event.preventDefault();
|
||||||
|
|
||||||
|
const taskId = document.getElementById('rework-task-id').value;
|
||||||
|
const comment = document.getElementById('rework-comment').value;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch(`/api/tasks/${taskId}/rework`, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
body: JSON.stringify({ comment })
|
||||||
|
});
|
||||||
|
|
||||||
|
if (response.ok) {
|
||||||
|
alert('Задача возвращена на доработку!');
|
||||||
|
closeReworkModal();
|
||||||
|
loadTasks();
|
||||||
|
loadActivityLogs();
|
||||||
|
} else {
|
||||||
|
const error = await response.json();
|
||||||
|
alert(error.error || 'Ошибка возврата задачи на доработку');
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка:', error);
|
||||||
|
alert('Ошибка возврата задачи на доработку');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function updateStatus(taskId, userId, status) {
|
||||||
|
try {
|
||||||
|
const response = await fetch(`/api/tasks/${taskId}/status`, {
|
||||||
|
method: 'PUT',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
body: JSON.stringify({ userId, status })
|
||||||
|
});
|
||||||
|
|
||||||
|
if (response.ok) {
|
||||||
|
loadTasks();
|
||||||
|
loadActivityLogs();
|
||||||
|
} else {
|
||||||
|
const error = await response.json();
|
||||||
|
alert(error.error || 'Ошибка обновления статуса');
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка:', error);
|
||||||
|
alert('Ошибка обновления статуса');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function canUserEditTask(task) {
|
||||||
|
if (!currentUser) return false;
|
||||||
|
|
||||||
|
// Администратор может всё
|
||||||
|
if (currentUser.role === 'admin') return true;
|
||||||
|
|
||||||
|
// Создатель может редактировать свою задачу
|
||||||
|
if (parseInt(task.created_by) === currentUser.id) {
|
||||||
|
// Но если задача уже назначена другим пользователям,
|
||||||
|
// создатель может только просматривать
|
||||||
|
if (task.assignments && task.assignments.length > 0) {
|
||||||
|
// Проверяем, назначена ли задача другим пользователям (не только себе)
|
||||||
|
const assignedToOthers = task.assignments.some(assignment =>
|
||||||
|
parseInt(assignment.user_id) !== currentUser.id
|
||||||
|
);
|
||||||
|
|
||||||
|
if (assignedToOthers) {
|
||||||
|
// Создатель может только просматривать и закрывать задачу
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Исполнитель может менять только свой статус
|
||||||
|
if (task.assignments) {
|
||||||
|
const isExecutor = task.assignments.some(assignment =>
|
||||||
|
parseInt(assignment.user_id) === currentUser.id
|
||||||
|
);
|
||||||
|
if (isExecutor) {
|
||||||
|
// Исполнитель может менять только статус
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
468
public/ui.js
Normal file
468
public/ui.js
Normal file
@@ -0,0 +1,468 @@
|
|||||||
|
// ui.js - UI функции и рендеринг
|
||||||
|
|
||||||
|
function showSection(sectionName) {
|
||||||
|
document.querySelectorAll('.section').forEach(section => {
|
||||||
|
section.classList.remove('active');
|
||||||
|
});
|
||||||
|
|
||||||
|
document.getElementById(sectionName + '-section').classList.add('active');
|
||||||
|
|
||||||
|
if (sectionName === 'tasks') {
|
||||||
|
loadTasks();
|
||||||
|
} else if (sectionName === 'logs') {
|
||||||
|
loadActivityLogs();
|
||||||
|
} else if (sectionName === 'kanban') {
|
||||||
|
loadKanbanTasks();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Загрузка профиля при переходе в личный кабинет
|
||||||
|
if (sectionName === 'profile') {
|
||||||
|
loadUserProfile();
|
||||||
|
loadNotificationSettings();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderTasks() {
|
||||||
|
const container = document.getElementById('tasks-list');
|
||||||
|
const showDeleted = document.getElementById('show-deleted')?.checked || false;
|
||||||
|
|
||||||
|
let filteredTasks = tasks;
|
||||||
|
if (!showDeleted) {
|
||||||
|
filteredTasks = tasks.filter(task => task.status === 'active');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (filteredTasks.length === 0) {
|
||||||
|
container.innerHTML = '<div class="loading">Задачи не найдены</div>';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
container.innerHTML = filteredTasks.map(task => {
|
||||||
|
const isExpanded = expandedTasks.has(task.id);
|
||||||
|
const overallStatus = getTaskOverallStatus(task);
|
||||||
|
const statusClass = getStatusClass(overallStatus);
|
||||||
|
const isDeleted = task.status === 'deleted';
|
||||||
|
const isClosed = task.closed_at !== null;
|
||||||
|
const userRole = getUserRoleInTask(task);
|
||||||
|
const canEdit = canUserEditTask(task);
|
||||||
|
const isCopy = task.original_task_id !== null;
|
||||||
|
|
||||||
|
const timeLeftInfo = getTimeLeftInfo(task);
|
||||||
|
|
||||||
|
return `
|
||||||
|
<div class="task-card ${isDeleted ? 'deleted' : ''} ${isClosed ? 'closed' : ''}" data-task-id="${task.id}">
|
||||||
|
<div class="task-header">
|
||||||
|
<div class="task-title" onclick="toggleTask(${task.id})" style="cursor: pointer; display: flex; justify-content: space-between; align-items: center;">
|
||||||
|
<div style="flex: 1;">
|
||||||
|
<span class="task-number">Задача №${task.id}</span>
|
||||||
|
<strong>${task.title}</strong>
|
||||||
|
${isDeleted ? '<span class="deleted-badge">Удалена</span>' : ''}
|
||||||
|
${isClosed ? '<span class="closed-badge">Закрыта</span>' : ''}
|
||||||
|
${isCopy ? '<span class="copy-badge">Копия</span>' : ''}
|
||||||
|
${timeLeftInfo ? `<span class="deadline-badge ${timeLeftInfo.class}">${timeLeftInfo.text}</span>` : ''}
|
||||||
|
<span class="role-badge ${getRoleBadgeClass(userRole)}">${userRole}</span>
|
||||||
|
</div>
|
||||||
|
<div class="task-status ${statusClass}">${getStatusText(overallStatus)}</div>
|
||||||
|
<div class="expand-icon" style="margin-left: 10px; transition: transform 0.3s; transform: rotate(${isExpanded ? '180deg' : '0deg'});">
|
||||||
|
▼
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="task-content ${isExpanded ? 'expanded' : ''}">
|
||||||
|
<div class="task-actions">
|
||||||
|
${!isDeleted && !isClosed ? `
|
||||||
|
${canEdit ? `<button class="edit-btn" onclick="openEditModal(${task.id})" title="Редактировать">✏️</button>` : ''}
|
||||||
|
<button class="copy-btn" onclick="openCopyModal(${task.id})" title="Создать копию">📋</button>
|
||||||
|
${canEdit ? `<button class="rework-btn" onclick="openReworkModal(${task.id})" title="Вернуть на доработку">🔄</button>` : ''}
|
||||||
|
${canEdit ? `<button class="close-btn" onclick="closeTask(${task.id})" title="Закрыть задачу">🔒</button>` : ''}
|
||||||
|
${canEdit ? `<button class="delete-btn" onclick="deleteTask(${task.id})" title="Удалить">🗑️</button>` : ''}
|
||||||
|
` : ''}
|
||||||
|
${isClosed && canEdit ? `
|
||||||
|
<button class="reopen-btn" onclick="reopenTask(${task.id})" title="Открыть задачу">🔓</button>
|
||||||
|
` : ''}
|
||||||
|
${isDeleted && currentUser.role === 'admin' ? `
|
||||||
|
<button class="restore-btn" onclick="restoreTask(${task.id})" title="Восстановить">↶</button>
|
||||||
|
` : ''}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
${isCopy && task.original_task_title ? `
|
||||||
|
<div class="task-original">
|
||||||
|
<small>Оригинал: "${task.original_task_title}" (создал: ${task.original_creator_name})</small>
|
||||||
|
</div>
|
||||||
|
` : ''}
|
||||||
|
|
||||||
|
<div class="task-description">${task.description || 'Нет описания'}</div>
|
||||||
|
|
||||||
|
${task.rework_comment ? `
|
||||||
|
<div class="rework-comment">
|
||||||
|
<strong>Комментарий к доработке:</strong> ${task.rework_comment}
|
||||||
|
</div>
|
||||||
|
` : ''}
|
||||||
|
|
||||||
|
<div class="task-dates-files">
|
||||||
|
<div class="task-dates">
|
||||||
|
<strong>Создана:</strong> ${formatDateTime(task.start_date || task.created_at)}
|
||||||
|
${task.due_date ? ` | <strong>Выполнить до:</strong> ${formatDateTime(task.due_date)}` : ''}
|
||||||
|
${showingTasksWithoutDate ? '<span class="no-date-badge">Без срока</span>' : ''}
|
||||||
|
</div>
|
||||||
|
<div class="file-list" id="files-${task.id}">
|
||||||
|
<strong>Файлы:</strong>
|
||||||
|
${task.files && task.files.length > 0 ?
|
||||||
|
`<div class="file-icons-container">${task.files.map(file => renderFileIcon(file)).join('')}</div>` :
|
||||||
|
'<span class="no-files">нет файлов</span>'
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="task-assignments">
|
||||||
|
<strong>Исполнители:</strong>
|
||||||
|
${task.assignments && task.assignments.length > 0 ?
|
||||||
|
renderAssignmentList(task.assignments, task.id, canEdit) :
|
||||||
|
'<div>Не назначены</div>'
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="task-meta">
|
||||||
|
<small>Создана: ${formatDateTime(task.created_at)} | Автор: ${task.creator_name}</small>
|
||||||
|
${task.deleted_at ? `<br><small>Удалена: ${formatDateTime(task.deleted_at)}</small>` : ''}
|
||||||
|
${task.closed_at ? `<br><small>Закрыта: ${formatDateTime(task.closed_at)}</small>` : ''}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}).join('');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Улучшенная функция рендеринга списка исполнителей с фильтрацией
|
||||||
|
function renderAssignmentList(assignments, taskId, canEdit) {
|
||||||
|
if (!assignments || assignments.length === 0) {
|
||||||
|
return '<div>Не назначены</div>';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Создаем контейнер с возможностью фильтрации
|
||||||
|
return `
|
||||||
|
<div class="assignments-container">
|
||||||
|
<div class="assignments-filter">
|
||||||
|
<input type="text"
|
||||||
|
class="assignment-filter-input"
|
||||||
|
placeholder="Поиск исполнителя..."
|
||||||
|
data-task-id="${taskId}"
|
||||||
|
oninput="filterAssignments(${taskId})">
|
||||||
|
<span class="filter-count" id="filter-count-${taskId}">${assignments.length} исполнителей</span>
|
||||||
|
</div>
|
||||||
|
<div class="assignments-scroll-container" id="assignments-${taskId}">
|
||||||
|
${assignments.map(assignment => renderAssignment(assignment, taskId, canEdit)).join('')}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Функция для фильтрации исполнителей в конкретной задаче
|
||||||
|
function filterAssignments(taskId) {
|
||||||
|
const filterInput = document.querySelector(`.assignment-filter-input[data-task-id="${taskId}"]`);
|
||||||
|
const scrollContainer = document.getElementById(`assignments-${taskId}`);
|
||||||
|
const filterCount = document.getElementById(`filter-count-${taskId}`);
|
||||||
|
|
||||||
|
if (!filterInput || !scrollContainer) return;
|
||||||
|
|
||||||
|
const searchTerm = filterInput.value.toLowerCase();
|
||||||
|
const assignments = scrollContainer.querySelectorAll('.assignment');
|
||||||
|
|
||||||
|
let visibleCount = 0;
|
||||||
|
|
||||||
|
assignments.forEach(assignment => {
|
||||||
|
const userName = assignment.querySelector('strong')?.textContent?.toLowerCase() || '';
|
||||||
|
const userLogin = assignment.querySelector('small')?.textContent?.toLowerCase() || '';
|
||||||
|
|
||||||
|
const isVisible = userName.includes(searchTerm) ||
|
||||||
|
userLogin.includes(searchTerm) ||
|
||||||
|
searchTerm === '';
|
||||||
|
|
||||||
|
assignment.style.display = isVisible ? '' : 'none';
|
||||||
|
|
||||||
|
if (isVisible) {
|
||||||
|
visibleCount++;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (filterCount) {
|
||||||
|
filterCount.textContent = `${visibleCount} из ${assignments.length} исполнителей`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggleTask(taskId) {
|
||||||
|
if (expandedTasks.has(taskId)) {
|
||||||
|
expandedTasks.delete(taskId);
|
||||||
|
} else {
|
||||||
|
expandedTasks.add(taskId);
|
||||||
|
loadTaskFiles(taskId);
|
||||||
|
}
|
||||||
|
renderTasks();
|
||||||
|
}
|
||||||
|
|
||||||
|
function getTimeLeftInfo(task) {
|
||||||
|
if (!task.due_date || task.closed_at) return null;
|
||||||
|
|
||||||
|
const dueDate = new Date(task.due_date);
|
||||||
|
const now = new Date();
|
||||||
|
const timeLeft = dueDate.getTime() - now.getTime();
|
||||||
|
const hoursLeft = Math.floor(timeLeft / (60 * 60 * 1000));
|
||||||
|
|
||||||
|
if (hoursLeft <= 0) return null;
|
||||||
|
|
||||||
|
if (hoursLeft <= 24) {
|
||||||
|
return {
|
||||||
|
text: `Менее 24ч`,
|
||||||
|
class: 'deadline-24h'
|
||||||
|
};
|
||||||
|
} else if (hoursLeft <= 48) {
|
||||||
|
return {
|
||||||
|
text: `Менее 48ч`,
|
||||||
|
class: 'deadline-48h'
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderAssignment(assignment, taskId, canEdit) {
|
||||||
|
const statusClass = getStatusClass(assignment.status);
|
||||||
|
const isCurrentUser = assignment.user_id === currentUser.id;
|
||||||
|
const isOverdue = assignment.status === 'overdue';
|
||||||
|
const isRework = assignment.status === 'rework';
|
||||||
|
|
||||||
|
const timeLeftInfo = getAssignmentTimeLeftInfo(assignment);
|
||||||
|
|
||||||
|
return `
|
||||||
|
<div class="assignment ${isOverdue ? 'overdue' : ''} ${isRework ? 'rework' : ''}">
|
||||||
|
<span class="assignment-status ${statusClass}"></span>
|
||||||
|
<div style="flex: 1;">
|
||||||
|
<strong>${assignment.user_name}</strong>
|
||||||
|
${isCurrentUser ? '<small>(Вы)</small>' : ''}
|
||||||
|
${timeLeftInfo ? `<span class="deadline-indicator ${timeLeftInfo.class}">${timeLeftInfo.text}</span>` : ''}
|
||||||
|
${assignment.start_date || assignment.due_date ? `
|
||||||
|
<div class="assignment-dates">
|
||||||
|
${assignment.start_date ? `<small>Начало: ${formatDateTime(assignment.start_date)}</small>` : ''}
|
||||||
|
${assignment.due_date ? `<small>Выполнить до: ${formatDateTime(assignment.due_date)}</small>` : ''}
|
||||||
|
</div>
|
||||||
|
` : ''}
|
||||||
|
${assignment.rework_comment ? `
|
||||||
|
<div class="assignment-rework-comment">
|
||||||
|
<small><strong>Комментарий:</strong> ${assignment.rework_comment}</small>
|
||||||
|
</div>
|
||||||
|
` : ''}
|
||||||
|
</div>
|
||||||
|
<div class="action-buttons">
|
||||||
|
${isCurrentUser && assignment.status === 'assigned' ?
|
||||||
|
`<button onclick="updateStatus(${taskId}, ${assignment.user_id}, 'in_progress')">Приступить</button>` : ''}
|
||||||
|
${isCurrentUser && (assignment.status === 'in_progress' || assignment.status === 'overdue' || assignment.status === 'rework') ?
|
||||||
|
`<button onclick="updateStatus(${taskId}, ${assignment.user_id}, 'completed')">Выполнено</button>` : ''}
|
||||||
|
${canEdit ?
|
||||||
|
`<button class="edit-date-btn" onclick="openEditAssignmentModal(${taskId}, ${assignment.user_id})" title="Редактировать сроки">📅</button>` : ''}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getAssignmentTimeLeftInfo(assignment) {
|
||||||
|
if (!assignment.due_date || assignment.status === 'completed') return null;
|
||||||
|
|
||||||
|
const dueDate = new Date(assignment.due_date);
|
||||||
|
const now = new Date();
|
||||||
|
const timeLeft = dueDate.getTime() - now.getTime();
|
||||||
|
const hoursLeft = Math.floor(timeLeft / (60 * 60 * 1000));
|
||||||
|
|
||||||
|
if (hoursLeft <= 0) return null;
|
||||||
|
|
||||||
|
if (hoursLeft <= 24) {
|
||||||
|
return {
|
||||||
|
text: `Осталось ${hoursLeft}ч`,
|
||||||
|
class: 'deadline-24h'
|
||||||
|
};
|
||||||
|
} else if (hoursLeft <= 48) {
|
||||||
|
return {
|
||||||
|
text: `Осталось ${hoursLeft}ч`,
|
||||||
|
class: 'deadline-48h'
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getTaskOverallStatus(task) {
|
||||||
|
if (task.status === 'deleted') return 'deleted';
|
||||||
|
if (task.closed_at) return 'closed';
|
||||||
|
if (!task.assignments || task.assignments.length === 0) return 'unassigned';
|
||||||
|
|
||||||
|
const assignments = task.assignments;
|
||||||
|
let hasAssigned = false;
|
||||||
|
let hasInProgress = false;
|
||||||
|
let hasOverdue = false;
|
||||||
|
let hasRework = false;
|
||||||
|
let allCompleted = true;
|
||||||
|
|
||||||
|
for (let assignment of assignments) {
|
||||||
|
if (assignment.status === 'assigned') {
|
||||||
|
hasAssigned = true;
|
||||||
|
allCompleted = false;
|
||||||
|
} else if (assignment.status === 'in_progress') {
|
||||||
|
hasInProgress = true;
|
||||||
|
allCompleted = false;
|
||||||
|
} else if (assignment.status === 'overdue') {
|
||||||
|
hasOverdue = true;
|
||||||
|
allCompleted = false;
|
||||||
|
} else if (assignment.status === 'rework') {
|
||||||
|
hasRework = true;
|
||||||
|
allCompleted = false;
|
||||||
|
} else if (assignment.status !== 'completed') {
|
||||||
|
allCompleted = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (allCompleted) return 'completed';
|
||||||
|
if (hasRework) return 'rework';
|
||||||
|
if (hasOverdue) return 'overdue';
|
||||||
|
if (hasInProgress) return 'in_progress';
|
||||||
|
if (hasAssigned) return 'assigned';
|
||||||
|
return 'unassigned';
|
||||||
|
}
|
||||||
|
|
||||||
|
function getStatusClass(status) {
|
||||||
|
switch (status) {
|
||||||
|
case 'deleted': return 'status-gray';
|
||||||
|
case 'closed': return 'status-gray';
|
||||||
|
case 'unassigned': return 'status-purple';
|
||||||
|
case 'assigned': return 'status-red';
|
||||||
|
case 'in_progress': return 'status-orange';
|
||||||
|
case 'rework': return 'status-yellow';
|
||||||
|
case 'overdue': return 'status-darkred';
|
||||||
|
case 'completed': return 'status-green';
|
||||||
|
default: return 'status-purple';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getStatusText(status) {
|
||||||
|
switch (status) {
|
||||||
|
case 'deleted': return 'Удалена';
|
||||||
|
case 'closed': return 'Закрыта';
|
||||||
|
case 'unassigned': return 'Не назначена';
|
||||||
|
case 'assigned': return 'Назначена';
|
||||||
|
case 'in_progress': return 'В работе';
|
||||||
|
case 'rework': return 'На доработке';
|
||||||
|
case 'overdue': return 'Просрочена';
|
||||||
|
case 'completed': return 'Выполнена';
|
||||||
|
default: return 'Неизвестно';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getUserRoleInTask(task) {
|
||||||
|
if (!currentUser) return 'Нет доступа';
|
||||||
|
|
||||||
|
if (currentUser.role === 'admin') return 'Администратор';
|
||||||
|
|
||||||
|
if (parseInt(task.created_by) === currentUser.id) {
|
||||||
|
if (task.assignments && task.assignments.length > 0) {
|
||||||
|
const assignedToOthers = task.assignments.some(assignment =>
|
||||||
|
parseInt(assignment.user_id) !== currentUser.id
|
||||||
|
);
|
||||||
|
if (assignedToOthers) {
|
||||||
|
return 'Создатель (только просмотр)';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 'Создатель';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (task.assignments) {
|
||||||
|
const isExecutor = task.assignments.some(assignment =>
|
||||||
|
parseInt(assignment.user_id) === currentUser.id
|
||||||
|
);
|
||||||
|
if (isExecutor) return 'Исполнитель';
|
||||||
|
}
|
||||||
|
|
||||||
|
return 'Наблюдатель';
|
||||||
|
}
|
||||||
|
|
||||||
|
function getRoleBadgeClass(role) {
|
||||||
|
switch (role) {
|
||||||
|
case 'Администратор': return 'role-admin';
|
||||||
|
case 'Заказчик': return 'role-creator';
|
||||||
|
case 'Исполнитель': return 'role-executor';
|
||||||
|
default: return '';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function formatDateTime(dateTimeString) {
|
||||||
|
if (!dateTimeString) return '';
|
||||||
|
const date = new Date(dateTimeString);
|
||||||
|
return date.toLocaleString('ru-RU');
|
||||||
|
}
|
||||||
|
|
||||||
|
function formatDateTimeForInput(dateTimeString) {
|
||||||
|
if (!dateTimeString) return '';
|
||||||
|
const date = new Date(dateTimeString);
|
||||||
|
return date.toISOString().slice(0, 16);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Логи активности
|
||||||
|
async function loadActivityLogs() {
|
||||||
|
try {
|
||||||
|
const response = await fetch('/api/activity-logs');
|
||||||
|
const logs = await response.json();
|
||||||
|
renderLogs(logs);
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка загрузки логов:', error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderLogs(logs) {
|
||||||
|
const container = document.getElementById('logs-list');
|
||||||
|
|
||||||
|
if (logs.length === 0) {
|
||||||
|
container.innerHTML = '<div class="loading">Логи не найдены</div>';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
container.innerHTML = logs.map(log => `
|
||||||
|
<div class="log-entry">
|
||||||
|
<div class="log-time">${formatDateTime(log.created_at)}</div>
|
||||||
|
<div><strong>${log.user_name}</strong> - ${getActionText(log.action)}</div>
|
||||||
|
<div>Задача: "${log.task_title}"</div>
|
||||||
|
${log.details ? `<div>Детали: ${log.details}</div>` : ''}
|
||||||
|
</div>
|
||||||
|
`).join('');
|
||||||
|
}
|
||||||
|
|
||||||
|
function getActionText(action) {
|
||||||
|
const actions = {
|
||||||
|
'TASK_CREATED': 'создал задачу',
|
||||||
|
'TASK_COPIED': 'создал копию задачи',
|
||||||
|
'TASK_UPDATED': 'обновил задачу',
|
||||||
|
'TASK_DELETED': 'удалил задачу',
|
||||||
|
'TASK_RESTORED': 'восстановил задачу',
|
||||||
|
'TASK_ASSIGNED': 'назначил задачу',
|
||||||
|
'TASK_ASSIGNMENTS_UPDATED': 'обновил назначения',
|
||||||
|
'ASSIGNMENT_UPDATED': 'обновил сроки исполнителя',
|
||||||
|
'STATUS_CHANGED': 'изменил статус задачи',
|
||||||
|
'FILE_UPLOADED': 'загрузил файл',
|
||||||
|
'FILE_COPIED': 'скопировал файл',
|
||||||
|
'TASK_SENT_FOR_REWORK': 'вернул задачу на доработку',
|
||||||
|
'TASK_CLOSED': 'закрыл задачу',
|
||||||
|
'TASK_REOPENED': 'открыл задачу'
|
||||||
|
};
|
||||||
|
|
||||||
|
return actions[action] || action;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Функция для просмотра деталей задачи
|
||||||
|
async function viewTaskDetails(taskId) {
|
||||||
|
try {
|
||||||
|
const response = await fetch(`/api/tasks/${taskId}`);
|
||||||
|
const task = await response.json();
|
||||||
|
|
||||||
|
// Можно открыть модальное окно с подробной информацией
|
||||||
|
// Или показать в отдельной секции
|
||||||
|
alert(`Задача: ${task.title}\n\nОписание: ${task.description || 'Нет описания'}\n\nСоздатель: ${task.creator_name}\nСрок: ${task.due_date ? new Date(task.due_date).toLocaleString('ru-RU') : 'Не установлен'}`);
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка загрузки деталей задачи:', error);
|
||||||
|
}
|
||||||
|
}
|
||||||
142
public/users.js
Normal file
142
public/users.js
Normal file
@@ -0,0 +1,142 @@
|
|||||||
|
// users.js - Управление пользователями
|
||||||
|
let users = [];
|
||||||
|
let allUsers = [];
|
||||||
|
let filteredUsers = [];
|
||||||
|
let selectedUsers = [];
|
||||||
|
let editSelectedUsers = [];
|
||||||
|
let copySelectedUsers = [];
|
||||||
|
|
||||||
|
async function loadUsers() {
|
||||||
|
try {
|
||||||
|
const response = await fetch('/api/users');
|
||||||
|
users = await response.json();
|
||||||
|
allUsers = users;
|
||||||
|
filteredUsers = [...users];
|
||||||
|
renderUsersChecklist();
|
||||||
|
renderEditUsersChecklist();
|
||||||
|
renderCopyUsersChecklist();
|
||||||
|
populateFilterDropdowns();
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Ошибка загрузки пользователей:', error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function populateFilterDropdowns() {
|
||||||
|
const creatorFilter = document.getElementById('creator-filter');
|
||||||
|
const assigneeFilter = document.getElementById('assignee-filter');
|
||||||
|
|
||||||
|
creatorFilter.innerHTML = '<option value="">Все заказчики</option>';
|
||||||
|
assigneeFilter.innerHTML = '<option value="">Все исполнители</option>';
|
||||||
|
|
||||||
|
users.forEach(user => {
|
||||||
|
const creatorOption = document.createElement('option');
|
||||||
|
creatorOption.value = user.id;
|
||||||
|
creatorOption.textContent = `${user.name} (${user.login})`;
|
||||||
|
creatorFilter.appendChild(creatorOption.cloneNode(true));
|
||||||
|
|
||||||
|
const assigneeOption = creatorOption.cloneNode(true);
|
||||||
|
assigneeFilter.appendChild(assigneeOption);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function filterUsers() {
|
||||||
|
const search = document.getElementById('user-search').value.toLowerCase();
|
||||||
|
filteredUsers = users.filter(user =>
|
||||||
|
user.name.toLowerCase().includes(search) ||
|
||||||
|
user.login.toLowerCase().includes(search) ||
|
||||||
|
user.email.toLowerCase().includes(search)
|
||||||
|
);
|
||||||
|
renderUsersChecklist();
|
||||||
|
}
|
||||||
|
|
||||||
|
function filterEditUsers() {
|
||||||
|
const search = document.getElementById('edit-user-search').value.toLowerCase();
|
||||||
|
const filtered = users.filter(user =>
|
||||||
|
user.name.toLowerCase().includes(search) ||
|
||||||
|
user.login.toLowerCase().includes(search) ||
|
||||||
|
user.email.toLowerCase().includes(search)
|
||||||
|
);
|
||||||
|
renderEditUsersChecklist(filtered);
|
||||||
|
}
|
||||||
|
|
||||||
|
function filterCopyUsers() {
|
||||||
|
const search = document.getElementById('copy-user-search').value.toLowerCase();
|
||||||
|
const filtered = users.filter(user =>
|
||||||
|
user.name.toLowerCase().includes(search) ||
|
||||||
|
user.login.toLowerCase().includes(search) ||
|
||||||
|
user.email.toLowerCase().includes(search)
|
||||||
|
);
|
||||||
|
renderCopyUsersChecklist(filtered);
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderUsersChecklist() {
|
||||||
|
const container = document.getElementById('users-checklist');
|
||||||
|
container.innerHTML = filteredUsers
|
||||||
|
.filter(user => user.id !== currentUser.id)
|
||||||
|
.map(user => `
|
||||||
|
<div class="checkbox-item">
|
||||||
|
<label>
|
||||||
|
<input type="checkbox" name="assignedUsers" value="${user.id}"
|
||||||
|
onchange="toggleUserSelection(this, ${user.id})">
|
||||||
|
${user.name} (${user.email})
|
||||||
|
${user.auth_type === 'ldap' ? '<small style="color: #666;"> - LDAP</small>' : ''}
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
`).join('');
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderEditUsersChecklist(filtered = users) {
|
||||||
|
const container = document.getElementById('edit-users-checklist');
|
||||||
|
container.innerHTML = filtered
|
||||||
|
.filter(user => user.id !== currentUser.id)
|
||||||
|
.map(user => `
|
||||||
|
<div class="checkbox-item">
|
||||||
|
<label>
|
||||||
|
<input type="checkbox" name="assignedUsers" value="${user.id}"
|
||||||
|
onchange="toggleEditUserSelection(this, ${user.id})">
|
||||||
|
${user.name} (${user.email})
|
||||||
|
${user.auth_type === 'ldap' ? '<small style="color: #666;"> - LDAP</small>' : ''}
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
`).join('');
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderCopyUsersChecklist(filtered = users) {
|
||||||
|
const container = document.getElementById('copy-users-checklist');
|
||||||
|
container.innerHTML = filtered
|
||||||
|
.filter(user => user.id !== currentUser.id)
|
||||||
|
.map(user => `
|
||||||
|
<div class="checkbox-item">
|
||||||
|
<label>
|
||||||
|
<input type="checkbox" name="assignedUsers" value="${user.id}"
|
||||||
|
onchange="toggleCopyUserSelection(this, ${user.id})">
|
||||||
|
${user.name} (${user.email})
|
||||||
|
${user.auth_type === 'ldap' ? '<small style="color: #666;"> - LDAP</small>' : ''}
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
`).join('');
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggleUserSelection(checkbox, userId) {
|
||||||
|
if (checkbox.checked) {
|
||||||
|
selectedUsers.push(userId);
|
||||||
|
} else {
|
||||||
|
selectedUsers = selectedUsers.filter(id => id !== userId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggleEditUserSelection(checkbox, userId) {
|
||||||
|
if (checkbox.checked) {
|
||||||
|
editSelectedUsers.push(userId);
|
||||||
|
} else {
|
||||||
|
editSelectedUsers = editSelectedUsers.filter(id => id !== userId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggleCopyUserSelection(checkbox, userId) {
|
||||||
|
if (checkbox.checked) {
|
||||||
|
copySelectedUsers.push(userId);
|
||||||
|
} else {
|
||||||
|
copySelectedUsers = copySelectedUsers.filter(id => id !== userId);
|
||||||
|
}
|
||||||
|
}
|
||||||
208
server.js
208
server.js
@@ -551,6 +551,214 @@ app.get('/admin', (req, res) => {
|
|||||||
res.sendFile(path.join(__dirname, 'public/admin.html'));
|
res.sendFile(path.join(__dirname, 'public/admin.html'));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// API для получения настроек уведомлений пользователя
|
||||||
|
app.get('/api/user/settings', requireAuth, async (req, res) => {
|
||||||
|
try {
|
||||||
|
if (!req.session.user || !req.session.user.id) {
|
||||||
|
return res.status(401).json({ error: 'Не аутентифицирован' });
|
||||||
|
}
|
||||||
|
|
||||||
|
const userId = req.session.user.id;
|
||||||
|
const { getDb } = require('./database');
|
||||||
|
const db = getDb();
|
||||||
|
|
||||||
|
db.get("SELECT email_notifications, notification_email, telegram_notifications, telegram_chat_id, vk_notifications, vk_user_id FROM user_settings WHERE user_id = ?",
|
||||||
|
[userId],
|
||||||
|
(err, settings) => {
|
||||||
|
if (err) {
|
||||||
|
console.error('❌ Ошибка получения настроек:', err);
|
||||||
|
return res.status(500).json({ error: 'Ошибка получения настроек' });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!settings) {
|
||||||
|
// Возвращаем настройки по умолчанию
|
||||||
|
res.json({
|
||||||
|
email_notifications: true,
|
||||||
|
notification_email: req.session.user.email || '',
|
||||||
|
telegram_notifications: false,
|
||||||
|
telegram_chat_id: '',
|
||||||
|
vk_notifications: false,
|
||||||
|
vk_user_id: ''
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
// Преобразуем boolean из SQLite (0/1) в true/false
|
||||||
|
const result = {
|
||||||
|
email_notifications: !!settings.email_notifications,
|
||||||
|
notification_email: settings.notification_email || '',
|
||||||
|
telegram_notifications: !!settings.telegram_notifications,
|
||||||
|
telegram_chat_id: settings.telegram_chat_id || '',
|
||||||
|
vk_notifications: !!settings.vk_notifications,
|
||||||
|
vk_user_id: settings.vk_user_id || ''
|
||||||
|
};
|
||||||
|
res.json(result);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
console.error('❌ Ошибка получения настроек:', error);
|
||||||
|
res.status(500).json({ error: 'Ошибка получения настроек' });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// API для сохранения настроек уведомлений
|
||||||
|
app.post('/api/user/settings', requireAuth, async (req, res) => {
|
||||||
|
try {
|
||||||
|
if (!req.session.user || !req.session.user.id) {
|
||||||
|
return res.status(401).json({ error: 'Не аутентифицирован' });
|
||||||
|
}
|
||||||
|
|
||||||
|
const userId = req.session.user.id;
|
||||||
|
const {
|
||||||
|
email_notifications,
|
||||||
|
notification_email,
|
||||||
|
telegram_notifications,
|
||||||
|
telegram_chat_id,
|
||||||
|
vk_notifications,
|
||||||
|
vk_user_id
|
||||||
|
} = req.body;
|
||||||
|
|
||||||
|
// Валидация
|
||||||
|
if (email_notifications === undefined ||
|
||||||
|
telegram_notifications === undefined ||
|
||||||
|
vk_notifications === undefined) {
|
||||||
|
return res.status(400).json({
|
||||||
|
error: 'Не все обязательные поля заполнены'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const { getDb } = require('./database');
|
||||||
|
const db = getDb();
|
||||||
|
|
||||||
|
// Проверяем, есть ли уже настройки для пользователя
|
||||||
|
db.get("SELECT id FROM user_settings WHERE user_id = ?", [userId], (err, existing) => {
|
||||||
|
if (err) {
|
||||||
|
console.error('❌ Ошибка проверки настроек:', err);
|
||||||
|
return res.status(500).json({ error: 'Ошибка сохранения настроек' });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (existing) {
|
||||||
|
// Обновляем существующие настройки
|
||||||
|
db.run(
|
||||||
|
`UPDATE user_settings SET
|
||||||
|
email_notifications = ?,
|
||||||
|
notification_email = ?,
|
||||||
|
telegram_notifications = ?,
|
||||||
|
telegram_chat_id = ?,
|
||||||
|
vk_notifications = ?,
|
||||||
|
vk_user_id = ?,
|
||||||
|
updated_at = CURRENT_TIMESTAMP
|
||||||
|
WHERE user_id = ?`,
|
||||||
|
[
|
||||||
|
email_notifications ? 1 : 0,
|
||||||
|
notification_email || '',
|
||||||
|
telegram_notifications ? 1 : 0,
|
||||||
|
telegram_chat_id || '',
|
||||||
|
vk_notifications ? 1 : 0,
|
||||||
|
vk_user_id || '',
|
||||||
|
userId
|
||||||
|
],
|
||||||
|
function(updateErr) {
|
||||||
|
if (updateErr) {
|
||||||
|
console.error('❌ Ошибка обновления настроек:', updateErr);
|
||||||
|
return res.status(500).json({ error: 'Ошибка сохранения настроек' });
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(`✅ Настройки пользователя ${userId} обновлены`);
|
||||||
|
res.json({ success: true });
|
||||||
|
}
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
// Создаем новые настройки
|
||||||
|
db.run(
|
||||||
|
`INSERT INTO user_settings
|
||||||
|
(user_id, email_notifications, notification_email,
|
||||||
|
telegram_notifications, telegram_chat_id,
|
||||||
|
vk_notifications, vk_user_id)
|
||||||
|
VALUES (?, ?, ?, ?, ?, ?, ?)`,
|
||||||
|
[
|
||||||
|
userId,
|
||||||
|
email_notifications ? 1 : 0,
|
||||||
|
notification_email || '',
|
||||||
|
telegram_notifications ? 1 : 0,
|
||||||
|
telegram_chat_id || '',
|
||||||
|
vk_notifications ? 1 : 0,
|
||||||
|
vk_user_id || ''
|
||||||
|
],
|
||||||
|
function(insertErr) {
|
||||||
|
if (insertErr) {
|
||||||
|
console.error('❌ Ошибка создания настроек:', insertErr);
|
||||||
|
return res.status(500).json({ error: 'Ошибка сохранения настроек' });
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(`✅ Настройки пользователя ${userId} созданы`);
|
||||||
|
res.json({ success: true });
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
console.error('❌ Ошибка сохранения настроек:', error);
|
||||||
|
res.status(500).json({ error: 'Ошибка сохранения настроек' });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// API для проверки настроек
|
||||||
|
app.get('/api/user/settings/check', requireAuth, (req, res) => {
|
||||||
|
try {
|
||||||
|
if (!req.session.user || !req.session.user.id) {
|
||||||
|
return res.status(401).json({ error: 'Не аутентифицирован' });
|
||||||
|
}
|
||||||
|
|
||||||
|
const userId = req.session.user.id;
|
||||||
|
const { getDb } = require('./database');
|
||||||
|
const db = getDb();
|
||||||
|
|
||||||
|
db.get("SELECT COUNT(*) as count FROM user_settings WHERE user_id = ?", [userId], (err, result) => {
|
||||||
|
if (err) {
|
||||||
|
console.error('❌ Ошибка проверки таблицы:', err);
|
||||||
|
return res.json({
|
||||||
|
table_exists: false,
|
||||||
|
user_has_settings: false,
|
||||||
|
error: err.message
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
res.json({
|
||||||
|
table_exists: true,
|
||||||
|
user_has_settings: result.count > 0,
|
||||||
|
user_id: userId
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
console.error('❌ Ошибка проверки настроек:', error);
|
||||||
|
res.status(500).json({ error: 'Ошибка проверки настроек' });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// API для проверки email уведомлений
|
||||||
|
app.get('/api/email-health', requireAuth, async (req, res) => {
|
||||||
|
try {
|
||||||
|
if (req.session.user.role !== 'admin') {
|
||||||
|
return res.status(403).json({ error: 'Недостаточно прав' });
|
||||||
|
}
|
||||||
|
|
||||||
|
const emailNotifications = require('./email-notifications');
|
||||||
|
const health = {
|
||||||
|
ready: emailNotifications.isReady(),
|
||||||
|
email: process.env.YANDEX_EMAIL,
|
||||||
|
host: process.env.YANDEX_SMTP_HOST,
|
||||||
|
timestamp: new Date().toISOString()
|
||||||
|
};
|
||||||
|
|
||||||
|
res.json(health);
|
||||||
|
} catch (error) {
|
||||||
|
res.status(500).json({ error: error.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
// Инициализация сервера
|
// Инициализация сервера
|
||||||
async function initializeServer() {
|
async function initializeServer() {
|
||||||
console.log('🚀 Инициализация сервера...');
|
console.log('🚀 Инициализация сервера...');
|
||||||
|
|||||||
Reference in New Issue
Block a user