переделал ui

This commit is contained in:
2026-03-04 12:09:40 +05:00
parent 3b68dfc042
commit 850a23e962
7 changed files with 1296 additions and 1685 deletions

View File

@@ -1,22 +1,31 @@
// Скрипт для отображения задач, где пользователь является автором
// loadMyCreatedTasks.js - Задачи, созданные пользователем и назначенные ему
// Глобальные переменные
let myAuthorTasks = [];
let myAuthorTasksFiltered = [];
let expandedMyTasks = new Set(); // Для отслеживания развернутых задач
let updateInterval = null; // Интервал обновления
let isUpdating = false; // Флаг для предотвращения множественных обновлений
let expandedMyTasks = new Set();
let updateInterval = null;
let isUpdating = false;
// Загрузка задач при открытии секции
// Показать секцию "Мои задачи (как автор)"
function showMyTasksSection() {
showSection('mytasks');
loadMyAuthorTasks();
startAutoUpdate(); // Запускаем автообновление при открытии секции
window.currentTaskView = 'my_assigned';
loadTasks();
startAutoUpdate();
}
// Показать секцию "Задачи для исполнения"
function showRunTasksSection() {
showSection('runtasks');
window.currentTaskView = 'assigned_to_me';
loadTasks();
startAutoUpdate();
}
// Остановка автообновления при уходе с секции
function hideMyTasksSection() {
function hideTasksSection() {
stopAutoUpdate();
}
// Запуск автоматического обновления
function startAutoUpdate() {
// Останавливаем предыдущий интервал, если был
@@ -29,6 +38,7 @@ function startAutoUpdate() {
console.log('🔄 Автообновление задач запущено (каждые 15 сек)');
}
// Остановка автоматического обновления
function stopAutoUpdate() {
if (updateInterval) {
@@ -37,6 +47,7 @@ function stopAutoUpdate() {
console.log('⏹️ Автообновление задач остановлено');
}
}
// Функция автоматического обновления
async function autoUpdateTasks() {
// Предотвращаем множественные обновления
@@ -47,7 +58,9 @@ async function autoUpdateTasks() {
// Проверяем, активна ли секция
const mytasksSection = document.getElementById('mytasks-section');
if (!mytasksSection || !mytasksSection.classList.contains('active')) {
const runtasksSection = document.getElementById('runtasks-section');
if ((!mytasksSection || !mytasksSection.classList.contains('active')) &&
(!runtasksSection || !runtasksSection.classList.contains('active'))) {
console.log('⏸️ Секция неактивна, автообновление приостановлено');
stopAutoUpdate();
return;
@@ -58,26 +71,10 @@ async function autoUpdateTasks() {
try {
console.log('🔄 Автообновление задач...', new Date().toLocaleTimeString());
const response = await fetch('/api/kanban-tasks?days=62&filter=created');
await loadTasks(); // просто перезагружаем с текущими фильтрами
if (!response.ok) {
throw new Error(`Ошибка сервера: ${response.status}`);
}
const data = await response.json();
const newTasks = data.tasks || [];
// Проверяем, изменились ли данные
if (hasTasksChanged(myAuthorTasks, newTasks)) {
console.log('📊 Данные изменились, обновляем отображение');
myAuthorTasks = newTasks;
filterMyTasks();
// Показываем уведомление об обновлении
showUpdateNotification();
} else {
console.log('📊 Данные не изменились');
}
// Показываем уведомление об обновлении
showUpdateNotification();
} catch (error) {
console.error('❌ Ошибка автообновления:', error);
@@ -85,6 +82,7 @@ async function autoUpdateTasks() {
isUpdating = false;
}
}
// Показ уведомления об обновлении
function showUpdateNotification() {
const notification = document.createElement('div');
@@ -108,140 +106,48 @@ function showUpdateNotification() {
}, 300);
}, 2000);
}
// Проверка, изменились ли данные
function hasTasksChanged(oldTasks, newTasks) {
if (oldTasks.length !== newTasks.length) return true;
// Создаем Map для быстрого сравнения
const oldMap = new Map(oldTasks.map(t => [t.id, t]));
for (const newTask of newTasks) {
const oldTask = oldMap.get(newTask.id);
if (!oldTask) return true;
// Проверяем важные поля
if (oldTask.kanbanStatus !== newTask.kanbanStatus) return true;
if (oldTask.title !== newTask.title) return true;
if (oldTask.description !== newTask.description) return true;
if (oldTask.due_date !== newTask.due_date) return true;
// Проверяем изменения в назначениях
if (!areAssignmentsEqual(oldTask.assignments, newTask.assignments)) return true;
// Проверяем изменения в файлах
if (!areFilesEqual(oldTask.files, newTask.files)) return true;
}
return false;
}
// Проверка равенства назначений
function areAssignmentsEqual(oldAssignments, newAssignments) {
if (!oldAssignments && !newAssignments) return true;
if (!oldAssignments || !newAssignments) return false;
if (oldAssignments.length !== newAssignments.length) return false;
const oldMap = new Map(oldAssignments.map(a => [a.user_id, a]));
for (const newAss of newAssignments) {
const oldAss = oldMap.get(newAss.user_id);
if (!oldAss) return false;
if (oldAss.status !== newAss.status) return false;
}
return true;
}
// Проверка равенства файлов
function areFilesEqual(oldFiles, newFiles) {
if (!oldFiles && !newFiles) return true;
if (!oldFiles || !newFiles) return false;
if (oldFiles.length !== newFiles.length) return false;
const oldIds = new Set(oldFiles.map(f => f.id));
const newIds = new Set(newFiles.map(f => f.id));
return oldIds.size === newIds.size &&
[...oldIds].every(id => newIds.has(id));
}
// Основная функция загрузки задач
async function loadMyAuthorTasks() {
const container = document.getElementById('mytasks-list');
try {
container.innerHTML = `
<div class="loading">
<div class="spinner"></div>
<p>Загрузка ваших задач...</p>
</div>
`;
const response = await fetch('/api/kanban-tasks?days=62&filter=created');
if (!response.ok) {
throw new Error(`Ошибка сервера: ${response.status}`);
}
const data = await response.json();
myAuthorTasks = data.tasks || [];
filterMyTasks();
} catch (error) {
console.error('Ошибка загрузки задач:', error);
container.innerHTML = `
<div class="loading">Ошибка загрузки задач: ${error.message}</div>
`;
}
}
// Функция фильтрации задач
// Функция фильтрации для "Мои задачи" - просто перезагружает с текущими фильтрами
function filterMyTasks() {
const statusFilter = document.getElementById('mytasks-status-filter')?.value || 'all';
const searchText = document.getElementById('mytasks-search')?.value.toLowerCase() || '';
myAuthorTasksFiltered = myAuthorTasks.filter(task => {
if (statusFilter !== 'all') {
const taskStatus = task.kanbanStatus || 'assigned';
if (taskStatus !== statusFilter) return false;
}
if (searchText) {
const title = task.title || '';
const description = task.description || '';
const searchable = `${title} ${description}`.toLowerCase();
if (!searchable.includes(searchText)) return false;
}
return true;
});
renderMyAuthorTasks();
loadTasks();
}
// Функция отображения задач в стиле ui.js
function renderMyAuthorTasks() {
// Функция фильтрации для "Задачи для исполнения"
function filterRunTasks() {
loadTasks();
}
// Рендеринг для "Мои задачи"
function renderMyTasks() {
const container = document.getElementById('mytasks-list');
if (!container) return;
if (myAuthorTasks.length === 0) {
if (!window.tasks || window.tasks.length === 0) {
container.innerHTML = '<div class="loading">У вас пока нет созданных задач</div>';
return;
}
if (myAuthorTasksFiltered.length === 0) {
container.innerHTML = '<div class="loading">Задачи не найдены</div>';
return;
// Используем общую функцию рендеринга, если она есть
if (typeof renderTasksInContainer === 'function') {
renderTasksInContainer('mytasks-list', window.tasks);
} else {
// Запасной вариант: рендерим прямо здесь (упрощённо)
renderMyTasksSimple(window.tasks);
}
}
// Упрощённый рендеринг для "Мои задачи" (если нет общей функции)
function renderMyTasksSimple(tasks) {
const container = document.getElementById('mytasks-list');
// Сортируем задачи по дате создания (новые сверху)
const sortedTasks = [...myAuthorTasksFiltered].sort((a, b) =>
const sortedTasks = [...tasks].sort((a, b) =>
new Date(b.created_at || 0) - new Date(a.created_at || 0)
);
container.innerHTML = sortedTasks.map(task => {
const isExpanded = expandedMyTasks.has(task.id);
const overallStatus = task.kanbanStatus || 'assigned';
const overallStatus = getTaskOverallStatus(task);
const statusClass = getStatusClass(overallStatus);
const isClosed = task.closed_at !== null;
const isCopy = task.original_task_id !== null;
@@ -317,7 +223,7 @@ function renderMyAuthorTasks() {
<div class="file-list" id="files-${task.id}">
<strong>Файлы:</strong>
${task.files && task.files.length > 0 ?
renderGroupedFilesWithDelete ? renderGroupedFilesWithDelete(task) : renderGroupedFiles(task)
(typeof renderGroupedFilesWithDelete === 'function' ? renderGroupedFilesWithDelete(task) : renderGroupedFiles(task))
: '<span class="no-files">нет файлов</span>'
}
</div>
@@ -346,7 +252,7 @@ function renderMyAuthorTasks() {
// Загружаем файлы для развернутых задач
expandedMyTasks.forEach(taskId => {
if (myAuthorTasks.some(t => t.id == taskId)) {
if (window.tasks.some(t => t.id == taskId)) {
loadTaskFiles(taskId);
}
});
@@ -360,388 +266,16 @@ function toggleMyTask(taskId) {
expandedMyTasks.add(taskId);
loadTaskFiles(taskId);
}
renderMyAuthorTasks();
renderMyTasks();
}
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 getTaskTypeDisplayName(type) {
const typeNames = {
'regular': 'Задача',
'document': 'Документ',
'it': 'ИТ',
'ahch': 'АХЧ',
'psychologist': 'Психолог',
'speech_therapist': 'Логопед',
'Social_educator': 'Социальный педагог',
'hr': 'Диспетчер расписания',
'certificate': 'Справка',
'e_journal': 'Эл. журнал'
};
return typeNames[type] || type;
}
function formatDateTime(dateTimeString) {
if (!dateTimeString) return '';
const date = new Date(dateTimeString);
return date.toLocaleString('ru-RU');
}
// Функция для рендеринга одного исполнителя (копия из ui.js с небольшими адаптациями)
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);
const task = myAuthorTasks.find(t => t.id === taskId);
const isTaskCreator = task && parseInt(task.created_by) === currentUser.id;
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>` : ''}
${isTaskCreator && assignment.status !== 'assigned' ?
`<button class="rework-btn" onclick="openReworkAssignmentModal(${taskId}, ${assignment.user_id}, '${assignment.user_name}')" title="Отправить на доработку">🔄Переделать</button>` : ''}
${isTaskCreator && assignment.status !== 'completed' ?
`<button class="force-complete-btn" onclick="forceCompleteAssignment(${taskId}, ${assignment.user_id}, '${assignment.user_name}')" title="Принудительно отметить как выполненное">✅ Завершить</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 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 openAddFileModal(taskId) {
if (typeof window.openAddFileModal === 'function') {
return window.openAddFileModal(taskId);
}
const task = myAuthorTasks.find(t => t.id === taskId);
if (!task) {
alert('Задача не найдена');
return;
}
const modalHtml = `
<div class="modal" id="add-file-modal">
<div class="modal-content">
<div class="modal-header">
<h3>Добавить файл к задаче: "${task.title}"</h3>
<span class="close" onclick="closeAddFileModal()">&times;</span>
</div>
<div class="modal-body">
<form id="add-file-form">
<input type="hidden" name="task_id" value="${taskId}">
<div class="form-group">
<label for="file-input">Выберите файл:</label>
<input type="file" id="file-input" name="files" multiple>
<small>Максимальный размер: 10MB</small>
</div>
<div class="form-group">
<label for="file-description">Описание файла (необязательно):</label>
<textarea id="file-description" name="description" rows="3"
placeholder="Описание файла..."></textarea>
</div>
<div class="modal-footer">
<button type="button" class="btn-cancel" onclick="closeAddFileModal()">Отмена</button>
<button type="submit" class="btn-primary">Добавить файл</button>
</div>
</form>
</div>
</div>
</div>
`;
const modalContainer = document.createElement('div');
modalContainer.innerHTML = modalHtml;
document.body.appendChild(modalContainer);
document.getElementById('add-file-form').addEventListener('submit', async function(e) {
e.preventDefault();
const fileInput = document.getElementById('file-input');
const description = document.getElementById('file-description').value;
if (fileInput.files.length === 0) {
alert('Выберите файл для загрузки');
return;
}
const file = fileInput.files[0];
const formData = new FormData();
formData.append('files', file);
formData.append('task_id', taskId);
if (description) {
formData.append('description', description);
}
try {
let response = await fetch(`/api/tasks/${taskId}/files`, {
method: 'POST',
body: formData
});
if (!response.ok) {
formData.delete('files');
formData.append('file', file);
response = await fetch(`/api/tasks/${taskId}/files`, {
method: 'POST',
body: formData
});
}
if (response.ok) {
alert('Файл успешно добавлен');
await loadTaskFiles(taskId);
closeAddFileModal();
if (expandedMyTasks.has(taskId)) {
renderMyAuthorTasks();
}
} else {
alert(`Ошибка при добавлении файла: ${response.status}`);
}
} catch (error) {
console.error('Ошибка:', error);
alert('Сетевая ошибка при добавлении файла');
}
});
setTimeout(() => {
document.getElementById('add-file-modal').style.display = 'block';
}, 10);
}
function closeAddFileModal() {
const modal = document.getElementById('add-file-modal');
if (modal) {
modal.style.display = 'none';
setTimeout(() => {
modal.parentElement.remove();
}, 300);
}
}
// Функция для открытия чата задачи
function openTaskChat(taskId) {
if (typeof window.openTaskChat === 'function') {
window.openTaskChat(taskId);
} else {
window.open(`/chat?task_id=${taskId}`, '_blank');
}
}
// Функция для открытия модального окна редактирования
function openEditModal(taskId) {
if (typeof window.openEditModal === 'function') {
window.openEditModal(taskId);
} else {
console.log('Открытие редактирования задачи:', taskId);
}
}
// Функция для открытия модального окна копирования
function openCopyModal(taskId) {
if (typeof window.openCopyModal === 'function') {
window.openCopyModal(taskId);
} else {
console.log('Открытие копирования задачи:', taskId);
}
}
// Функция для открытия модального окна доработки
function openReworkModal(taskId) {
if (typeof window.openReworkModal === 'function') {
window.openReworkModal(taskId);
} else {
console.log('Открытие доработки задачи:', taskId);
}
}
// Функция для закрытия задачи
async function closeTask(taskId) {
if (!confirm('Вы уверены, что хотите закрыть задачу?')) return;
try {
const response = await fetch(`/api/tasks/${taskId}/close`, {
method: 'PUT'
});
if (response.ok) {
alert('Задача закрыта');
loadMyAuthorTasks();
} 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('Задача удалена');
loadMyAuthorTasks();
} else {
const error = await response.json();
alert(`Ошибка: ${error.error || 'Неизвестная ошибка'}`);
}
} catch (error) {
console.error('Ошибка:', error);
alert('Сетевая ошибка при удалении задачи');
}
}
// Функция для обновления статуса исполнителя
async function updateStatus(taskId, userId, newStatus) {
try {
const response = await fetch(`/api/tasks/${taskId}/assignments/${userId}`, {
method: 'PUT',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({ status: newStatus })
});
if (response.ok) {
loadMyAuthorTasks();
} else {
const error = await response.json();
alert(`Ошибка: ${error.error || 'Неизвестная ошибка'}`);
}
} catch (error) {
console.error('Ошибка:', error);
alert('Сетевая ошибка при обновлении статуса');
}
}
// Автоматическая загрузка при открытии секции
document.addEventListener('DOMContentLoaded', () => {
const mytasksSection = document.getElementById('mytasks-section');
if (mytasksSection && mytasksSection.style.display !== 'none') {
loadMyAuthorTasks();
}
});
// Остальные вспомогательные функции (getStatusClass, getTaskTypeDisplayName, formatDateTime, renderAssignment, filterAssignments, openAddFileModal, closeAddFileModal, openTaskChat, openEditModal, openCopyModal, openReworkModal, closeTask, deleteTask, updateStatus) остаются без изменений
// ...
// Экспортируем функции в глобальную область
window.showMyTasksSection = showMyTasksSection;
window.loadMyAuthorTasks = loadMyAuthorTasks;
window.showRunTasksSection = showRunTasksSection;
window.filterMyTasks = filterMyTasks;
window.filterRunTasks = filterRunTasks;
window.toggleMyTask = toggleMyTask;
window.openAddFileModal = openAddFileModal;
window.closeAddFileModal = closeAddFileModal;
window.openTaskChat = openTaskChat;
window.openEditModal = openEditModal;
window.openCopyModal = openCopyModal;
window.openReworkModal = openReworkModal;
window.closeTask = closeTask;
window.deleteTask = deleteTask;
window.updateStatus = updateStatus;
window.filterAssignments = filterAssignments;
window.renderGroupedFiles = renderGroupedFiles;
window.renderMyTasks = renderMyTasks;