Compare commits

...

4 Commits

Author SHA1 Message Date
2e0dc90220 Merge branch 'master' of github.com:smartsoltech/smartsoltech.kr 2025-11-25 06:39:14 +09:00
1d193a9eab Add deployment scripts and documentation 2025-11-25 06:39:04 +09:00
Andrew K. Tsoy
67aeac2238 Merge pull request #2 from smartsoltech/frontend-redesign
merge
2025-11-25 06:38:02 +09:00
b1759eac10 merge 2025-11-24 08:10:56 +09:00
11 changed files with 849 additions and 0 deletions

157
SCRIPTS_README.md Normal file
View File

@@ -0,0 +1,157 @@
# SmartSolTech - Скрипты управления проектом
Этот набор скриптов предназначен для автоматизации процессов разработки и развертывания проекта SmartSolTech.
## Доступные скрипты
### 🚀 `./update.sh` - Полное обновление проекта
Выполняет полный цикл обновления:
- Создание бэкапа в удаленном репозитории
- Обновление кода из Git
- Остановка текущих контейнеров
- Пересборка Docker образов
- Запуск новых контейнеров
- Выполнение миграций Django
- Сбор статических файлов
- Проверка здоровья сервисов
**Использование:**
```bash
./update.sh # Полное обновление
./update.sh --help # Показать справку
./update.sh --logs # Показать логи без обновления
./update.sh --status # Показать статус сервисов
```
### ▶️ `./start.sh` - Быстрый запуск
Быстро запускает все сервисы проекта:
- Запуск Docker контейнеров
- Проверка статуса сервисов
- Отображение доступных ресурсов
**Использование:**
```bash
./start.sh
```
### ⏹️ `./stop.sh` - Остановка сервисов
Останавливает сервисы с различными опциями:
- Простая остановка контейнеров
- Остановка с удалением контейнеров и волюмов
- Полная очистка (включая образы)
**Использование:**
```bash
./stop.sh # Простая остановка
./stop.sh --remove # Остановка + удаление контейнеров и волюмов
./stop.sh --clean # Полная очистка (контейнеры + образы + волюмы)
./stop.sh --help # Показать справку
```
### 📋 `./logs.sh` - Просмотр логов
Показывает логи сервисов в реальном времени:
- Все логи или конкретного сервиса
- Последние N строк логов
- Интерактивный режим
**Использование:**
```bash
./logs.sh # Все логи в реальном времени
./logs.sh web # Логи только веб-сервера
./logs.sh db # Логи базы данных
./logs.sh pgadmin # Логи PgAdmin
./logs.sh --tail 50 # Последние 50 строк
./logs.sh --help # Показать справку
```
## Доступные сервисы
После запуска будут доступны:
- **Веб-сайт**: http://localhost:8000
- **Админка Django**: http://localhost:8000/admin
- **PgAdmin**: http://localhost:8080
## Требования
Убедитесь что установлены:
- Git
- Docker
- Docker Compose
- curl (для проверки здоровья)
## Безопасность
- Скрипт автоматически создает бэкапы перед обновлением
- Локальные изменения сохраняются в Git stash
- Staticfiles автоматически очищаются для избежания конфликтов
- Проверка статуса сервисов после запуска
## Примеры использования
### Ежедневное обновление
```bash
# Полное обновление с проверкой
./update.sh
# Если что-то пошло не так - смотрим логи
./logs.sh --tail 100
```
### Разработка
```bash
# Запуск для разработки
./start.sh
# Просмотр логов во время разработки
./logs.sh web
# Остановка после работы
./stop.sh
```
### Полная перезагрузка
```bash
# Полная очистка и пересборка
./stop.sh --clean
./update.sh
```
## Troubleshooting
### Проблемы с контейнерами
```bash
# Проверить статус
./logs.sh --help
docker-compose ps
# Перезапустить проблемный сервис
docker-compose restart web
```
### Проблемы с базой данных
```bash
# Проверить логи БД
./logs.sh db
# Принудительная пересборка
./stop.sh --remove
./update.sh
```
### Проблемы с Git
```bash
# Проверить статус
git status
# Восстановить из stash если нужно
git stash list
git stash apply
```
## Настройка автоматического обновления
Добавьте в crontab для автоматического обновления:
```bash
# Обновление каждый день в 3:00 утра
0 3 * * * cd /path/to/project && ./update.sh >> /var/log/smartsoltech-update.log 2>&1
```

62
loading_screen_fixed.html Normal file
View File

@@ -0,0 +1,62 @@
<!DOCTYPE html>
<html lang="ru">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Тест исправленного сайта SmartSolTech</title>
<style>
body { font-family: Arial, sans-serif; padding: 20px; line-height: 1.6; }
.success { background: #d4edda; padding: 15px; border-radius: 5px; margin: 10px 0; }
.info { background: #d1ecf1; padding: 15px; border-radius: 5px; margin: 10px 0; }
.test-link { display: inline-block; margin: 10px; padding: 10px 15px; background: #007bff; color: white; text-decoration: none; border-radius: 5px; }
.test-link:hover { background: #0056b3; }
</style>
</head>
<body>
<h1>🎉 Проблема с загрузочным экраном исправлена!</h1>
<div class="success">
<h2>✅ Что было исправлено:</h2>
<ul>
<li><strong>Синтаксические ошибки JavaScript:</strong> Исправлены проблемы с закрывающими скобками</li>
<li><strong>Загрузочный экран:</strong> Теперь корректно скрывается через 0.5 секунды</li>
<li><strong>Основная функциональность:</strong> Восстановлена работа тем, навигации и анимаций</li>
<li><strong>Отладочный код:</strong> Удален лишний код, который вызывал проблемы</li>
</ul>
</div>
<div class="info">
<h2>🧪 Тестирование:</h2>
<p>Теперь сайт должен нормально загружаться. Проверьте следующие страницы:</p>
<a href="http://localhost:8000/" target="_blank" class="test-link">🏠 Главная страница</a>
<a href="http://localhost:8000/services/" target="_blank" class="test-link">🛠️ Страница услуг</a>
<a href="http://localhost:8000/about/" target="_blank" class="test-link"> О нас</a>
<h3>Что проверить:</h3>
<ul>
<li>✅ Загрузочный экран исчезает через полсекунды</li>
<li>✅ Навигационные ссылки работают</li>
<li>✅ Кнопки и интерактивные элементы кликабельны</li>
<li>✅ Переключатель темы работает (если есть)</li>
<li>✅ Модальное окно услуг открывается</li>
<li>✅ QR-код генерация работает в модальном окне</li>
</ul>
</div>
<div class="info">
<h2>🛠️ Техническая информация:</h2>
<ul>
<li><strong>Проблема:</strong> Синтаксические ошибки в modern-scripts.js блокировали загрузку</li>
<li><strong>Решение:</strong> Пересоздан упрощенный и стабильный JavaScript файл</li>
<li><strong>Сохранена функциональность:</strong> Скрытие загрузочного экрана, переключение тем, плавная прокрутка, анимации</li>
<li><strong>Удалено:</strong> Отладочный код, который вызывал конфликты</li>
</ul>
</div>
<div class="success">
<h2>🚀 Сайт готов к использованию!</h2>
<p>Все основные функции восстановлены, включая механизм QR-кода для заявок через Telegram бота.</p>
</div>
</body>
</html>

View File

@@ -0,0 +1,110 @@
<!DOCTYPE html>
<html lang="ru">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>✨ Новая анимация успеха для заявок</title>
<style>
body { font-family: Arial, sans-serif; padding: 20px; line-height: 1.6; }
.improvement { background: #e8f5e8; padding: 20px; border-radius: 10px; margin: 15px 0; border-left: 5px solid #28a745; }
.feature { background: #e7f3ff; padding: 15px; border-radius: 8px; margin: 10px 0; }
.workflow { background: #fff9e6; padding: 15px; border-radius: 8px; margin: 10px 0; }
.test-btn { display: inline-block; padding: 12px 20px; background: #007bff; color: white; text-decoration: none; border-radius: 5px; margin: 10px 5px; }
.test-btn:hover { background: #0056b3; text-decoration: none; color: white; }
h2 { color: #333; border-bottom: 2px solid #28a745; padding-bottom: 5px; }
.emoji { font-size: 1.2em; }
</style>
</head>
<body>
<h1><span class="emoji"></span> Улучшенная анимация подачи заявок</h1>
<div class="improvement">
<h2><span class="emoji">🎯</span> Что улучшено:</h2>
<div class="feature">
<h3><span class="emoji">📐</span> QR-код отцентрирован</h3>
<ul>
<li>QR-код теперь идеально выровнен по центру поля</li>
<li>Фиксированный размер 200x200px для консистентности</li>
<li>Красивая рамка и отступы</li>
</ul>
</div>
<div class="feature">
<h3><span class="emoji">🎬</span> Анимированная галочка успеха</h3>
<ul>
<li>После создания заявки появляется анимированная галочка ✓</li>
<li>Плавная анимация рисования круга и галочки</li>
<li>Профессиональный зеленый дизайн</li>
</ul>
</div>
<div class="feature">
<h3><span class="emoji">⏱️</span> Автоматическое закрытие</h3>
<ul>
<li>Модальное окно автоматически закрывается через 6 секунд</li>
<li>3 сек на QR-код → 3 сек на анимацию успеха → закрытие</li>
<li>Уведомление о успешной подаче заявки</li>
</ul>
</div>
</div>
<div class="workflow">
<h2><span class="emoji">🔄</span> Новый процесс подачи заявки:</h2>
<ol>
<li><strong>Пользователь заполняет форму</strong> и нажимает "Отправить заявку"</li>
<li><strong>Показывается индикатор загрузки</strong> "Отправляем..."</li>
<li><strong>Появляется центрированный QR-код</strong> с кнопкой "Открыть в Telegram"</li>
<li><strong>Через 3 секунды</strong> QR-код скрывается</li>
<li><strong>Появляется анимированная галочка</strong> с сообщением "Заявка подана успешно!"</li>
<li><strong>Через 3 секунды</strong> модальное окно автоматически закрывается</li>
<li><strong>Показывается уведомление</strong> "Ожидайте подтверждения в Telegram"</li>
</ol>
</div>
<div class="improvement">
<h2><span class="emoji">🧪</span> Тестирование:</h2>
<p>Перейдите на страницу услуг и попробуйте подать заявку:</p>
<a href="http://localhost:8000/services/" target="_blank" class="test-btn">
<span class="emoji">🛠️</span> Тестировать на странице услуг
</a>
<h3>Что тестировать:</h3>
<ul>
<li>✅ Заполните форму и отправьте заявку</li>
<li>✅ Убедитесь, что QR-код отцентрирован</li>
<li>✅ Подождите 3 секунды - должна появиться анимированная галочка</li>
<li>✅ Подождите еще 3 секунды - окно должно закрыться автоматически</li>
<li>✅ Должно появиться уведомление о успешной подаче</li>
</ul>
</div>
<div class="feature">
<h2><span class="emoji">💻</span> Технические детали:</h2>
<h3>Добавленные компоненты:</h3>
<ul>
<li><strong>CSS анимации:</strong> Keyframes для рисования галочки и круга</li>
<li><strong>JavaScript таймеры:</strong> Координация показа QR → анимации → закрытия</li>
<li><strong>Центрирование:</strong> Flexbox для идеального выравнивания</li>
<li><strong>Сброс состояния:</strong> Очистка всех секций при закрытии модального окна</li>
</ul>
<h3>Файлы изменены:</h3>
<ul>
<li><code>services_modern.html</code> - обновлена структура модального окна</li>
<li><code>modern-styles.css</code> - добавлены анимации галочки</li>
</ul>
</div>
<div class="improvement">
<h2><span class="emoji">🎉</span> Результат:</h2>
<p><strong>Теперь процесс подачи заявки стал более интуитивным, визуально привлекательным и автоматизированным!</strong></p>
<p>Пользователи получают четкую обратную связь на каждом этапе, а модальное окно автоматически закрывается, предотвращая накопление открытых окон.</p>
</div>
</body>
</html>

View File

@@ -0,0 +1,358 @@
// Modern Scripts for SmartSolTech Website
document.addEventListener('DOMContentLoaded', function() {
console.log('SmartSolTech: DOM loaded, initializing...');
// Hide loading screen
const loadingScreen = document.getElementById('loading-screen');
if (loadingScreen) {
console.log('SmartSolTech: Loading screen found, hiding...');
setTimeout(() => {
loadingScreen.style.opacity = '0';
loadingScreen.style.pointerEvents = 'none';
setTimeout(() => {
loadingScreen.style.display = 'none';
loadingScreen.remove(); // Полностью удаляем элемент
console.log('SmartSolTech: Loading screen removed');
}, 300);
}, 500); // Уменьшили время ожидания
} else {
console.log('SmartSolTech: Loading screen not found');
}
// Theme Toggle Functionality
const themeToggle = document.getElementById('theme-toggle');
const html = document.documentElement;
// Check for saved theme preference
const currentTheme = localStorage.getItem('theme') || 'light';
html.setAttribute('data-theme', currentTheme);
updateThemeIcon(currentTheme);
themeToggle.addEventListener('click', function() {
const currentTheme = html.getAttribute('data-theme');
const newTheme = currentTheme === 'dark' ? 'light' : 'dark';
html.setAttribute('data-theme', newTheme);
localStorage.setItem('theme', newTheme);
updateThemeIcon(newTheme);
// Add animation
this.style.transform = 'scale(0.8)';
setTimeout(() => {
this.style.transform = 'scale(1)';
}, 150);
});
function updateThemeIcon(theme) {
const icon = themeToggle.querySelector('i');
if (theme === 'dark') {
icon.className = 'fas fa-sun';
themeToggle.setAttribute('aria-label', 'Переключить на светлую тему');
} else {
icon.className = 'fas fa-moon';
themeToggle.setAttribute('aria-label', 'Переключить на темную тему');
}
}
// Navbar scroll behavior
const navbar = document.querySelector('.navbar-modern');
let lastScrollTop = 0;
window.addEventListener('scroll', function() {
const scrollTop = window.pageYOffset || document.documentElement.scrollTop;
// Add/remove scrolled class
if (scrollTop > 50) {
navbar.classList.add('scrolled');
} else {
navbar.classList.remove('scrolled');
}
// Hide/show navbar on scroll
if (scrollTop > lastScrollTop && scrollTop > 100) {
navbar.style.transform = 'translateY(-100%)';
} else {
navbar.style.transform = 'translateY(0)';
}
lastScrollTop = scrollTop;
});
// Scroll to top button
const scrollToTopBtn = document.getElementById('scroll-to-top');
window.addEventListener('scroll', function() {
if (window.pageYOffset > 300) {
scrollToTopBtn.style.display = 'block';
scrollToTopBtn.style.opacity = '1';
} else {
scrollToTopBtn.style.opacity = '0';
setTimeout(() => {
if (window.pageYOffset <= 300) {
scrollToTopBtn.style.display = 'none';
}
}, 300);
}
});
scrollToTopBtn.addEventListener('click', function() {
window.scrollTo({
top: 0,
behavior: 'smooth'
});
});
// Smooth scrolling for anchor links
document.querySelectorAll('a[href^="#"]').forEach(anchor => {
anchor.addEventListener('click', function(e) {
e.preventDefault();
const target = document.querySelector(this.getAttribute('href'));
if (target) {
const offsetTop = target.offsetTop - 80; // Account for fixed navbar
window.scrollTo({
top: offsetTop,
behavior: 'smooth'
});
}
});
});
// Intersection Observer for animations
const observerOptions = {
threshold: 0.1,
rootMargin: '0px 0px -50px 0px'
};
const observer = new IntersectionObserver(function(entries) {
entries.forEach(entry => {
if (entry.isIntersecting) {
entry.target.classList.add('animate-fade-in-up');
// Add stagger delay for child elements
const children = entry.target.querySelectorAll('.service-card, .feature-list > *, .step-card');
children.forEach((child, index) => {
setTimeout(() => {
child.classList.add('animate-fade-in-up');
}, index * 100);
});
}
});
}, observerOptions);
// Observe elements for animation
document.querySelectorAll('.service-card, .card-modern, .step-card').forEach(el => {
observer.observe(el);
});
// Form enhancements
const forms = document.querySelectorAll('form');
forms.forEach(form => {
form.addEventListener('submit', function(e) {
const submitBtn = form.querySelector('button[type="submit"]');
if (submitBtn) {
const originalContent = submitBtn.innerHTML;
submitBtn.innerHTML = '<i class="fas fa-spinner fa-spin me-2"></i>Отправляем...';
submitBtn.disabled = true;
// Re-enable after 3 seconds (in case of slow response)
setTimeout(() => {
submitBtn.innerHTML = originalContent;
submitBtn.disabled = false;
}, 3000);
}
});
});
// Parallax effect for hero section
window.addEventListener('scroll', function() {
const scrolled = window.pageYOffset;
const parallaxElements = document.querySelectorAll('.animate-float');
parallaxElements.forEach(element => {
const speed = 0.5;
element.style.transform = `translateY(${scrolled * speed}px)`;
});
});
// Service cards hover effect
document.querySelectorAll('.service-card').forEach(card => {
card.addEventListener('mouseenter', function() {
this.style.transform = 'translateY(-10px) scale(1.02)';
});
card.addEventListener('mouseleave', function() {
this.style.transform = 'translateY(0) scale(1)';
});
});
// Card modern hover effects
document.querySelectorAll('.card-modern').forEach(card => {
card.addEventListener('mouseenter', function() {
this.style.boxShadow = '0 25px 50px -12px rgba(0, 0, 0, 0.25)';
});
card.addEventListener('mouseleave', function() {
this.style.boxShadow = 'var(--shadow)';
});
});
// Add loading animation to buttons
document.querySelectorAll('.btn-primary-modern, .btn-secondary-modern').forEach(btn => {
btn.addEventListener('click', function(e) {
// Create ripple effect
const ripple = document.createElement('span');
const rect = this.getBoundingClientRect();
const size = Math.max(rect.width, rect.height);
const x = e.clientX - rect.left - size / 2;
const y = e.clientY - rect.top - size / 2;
ripple.style.cssText = `
position: absolute;
border-radius: 50%;
background: rgba(255, 255, 255, 0.4);
transform: scale(0);
animation: ripple 0.6s linear;
width: ${size}px;
height: ${size}px;
left: ${x}px;
top: ${y}px;
`;
this.style.position = 'relative';
this.style.overflow = 'hidden';
this.appendChild(ripple);
setTimeout(() => {
ripple.remove();
}, 600);
});
});
// Typing animation for hero text (optional)
const typingText = document.querySelector('.typing-text');
if (typingText) {
const text = typingText.textContent;
typingText.textContent = '';
let i = 0;
function typeWriter() {
if (i < text.length) {
typingText.textContent += text.charAt(i);
i++;
setTimeout(typeWriter, 100);
}
}
setTimeout(typeWriter, 1000);
}
// Mobile menu enhancements
const navbarToggler = document.querySelector('.navbar-toggler');
const navbarCollapse = document.querySelector('.navbar-collapse');
if (navbarToggler && navbarCollapse) {
navbarToggler.addEventListener('click', function() {
const isExpanded = this.getAttribute('aria-expanded') === 'true';
// Animate the toggler icon
this.style.transform = 'rotate(180deg)';
setTimeout(() => {
this.style.transform = 'rotate(0deg)';
}, 300);
});
// Close menu when clicking on a link
document.querySelectorAll('.navbar-nav .nav-link').forEach(link => {
link.addEventListener('click', () => {
const bsCollapse = new bootstrap.Collapse(navbarCollapse, {
hide: true
});
});
});
}
// Newsletter form
const newsletterForm = document.querySelector('footer form');
if (newsletterForm) {
newsletterForm.addEventListener('submit', function(e) {
e.preventDefault();
const email = this.querySelector('input[type="email"]').value;
if (email) {
// Show success message
const button = this.querySelector('button');
const originalContent = button.innerHTML;
button.innerHTML = '<i class="fas fa-check"></i>';
button.style.background = '#10b981';
setTimeout(() => {
button.innerHTML = originalContent;
button.style.background = '';
this.reset();
}, 2000);
}
});
}
});
// Add CSS for ripple animation
const style = document.createElement('style');
style.textContent = `
@keyframes ripple {
to {
transform: scale(2);
opacity: 0;
}
}
.animate-fade-in-up {
opacity: 1 !important;
transform: translateY(0) !important;
}
/* Smooth transitions */
.navbar-modern {
transition: transform 0.3s ease, background-color 0.3s ease;
}
.service-card, .card-modern {
opacity: 0;
transform: translateY(30px);
transition: all 0.6s ease;
}
.step-card {
opacity: 0;
transform: translateX(-30px);
transition: all 0.6s ease;
}
.step-card:nth-child(even) {
transform: translateX(30px);
}
`;
document.head.appendChild(style);
// Отладка кликов
document.addEventListener('click', function(event) {
console.log('SmartSolTech: Click detected on:', event.target);
// Проверяем, не блокируются ли клики
const computedStyle = getComputedStyle(event.target);
if (computedStyle.pointerEvents === 'none') {
console.warn('SmartSolTech: Element has pointer-events: none');
}
// Проверяем z-index элементов
let element = event.target;
while (element && element !== document.body) {
const style = getComputedStyle(element);
if (style.zIndex && style.zIndex !== 'auto') {
console.log('SmartSolTech: Element z-index:', element.tagName, style.zIndex);
}
element = element.parentElement;
}
});
console.log('SmartSolTech: All scripts loaded successfully');
});
});

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB

35
test_links.html Normal file
View File

@@ -0,0 +1,35 @@
<!DOCTYPE html>
<html lang="ru">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Тест ссылок SmartSolTech</title>
</head>
<body>
<h1>Тестирование ссылок SmartSolTech</h1>
<div>
<h3>Основные страницы:</h3>
<ul>
<li><a href="http://localhost:8000/" target="_blank">Главная</a></li>
<li><a href="http://localhost:8000/services/" target="_blank">Услуги</a></li>
<li><a href="http://localhost:8000/about/" target="_blank">О нас</a></li>
</ul>
<p><strong>Инструкция:</strong></p>
<ol>
<li>Откройте каждую ссылку в новой вкладке</li>
<li>Проверьте, загружается ли страница</li>
<li>Попробуйте нажать на ссылки в навигации на каждой странице</li>
<li>Проверьте кнопки и интерактивные элементы</li>
</ol>
<p><strong>Что проверить на каждой странице:</strong></p>
<ul>
<li>Работают ли ссылки в навигационном меню</li>
<li>Работают ли кнопки (например, "Оставить заявку")</li>
<li>Исчезает ли загрузочный экран через несколько секунд</li>
<li>Работает ли переключатель темы</li>
</ul>
</div>
</body>
</html>

127
test_qr_functionality.html Normal file
View File

@@ -0,0 +1,127 @@
<!DOCTYPE html>
<html lang="ru">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Тестирование QR-кода для заявок</title>
<style>
body { font-family: Arial, sans-serif; padding: 20px; line-height: 1.6; }
.section { margin-bottom: 30px; padding: 20px; border: 1px solid #ddd; border-radius: 8px; }
.instructions { background: #f8f9fa; }
.test-case { background: #e7f3ff; }
.success { background: #d4edda; }
.warning { background: #fff3cd; }
h2 { color: #333; border-bottom: 2px solid #007bff; padding-bottom: 5px; }
h3 { color: #666; }
.step { margin: 10px 0; padding: 10px; background: white; border-radius: 4px; }
.link { color: #007bff; text-decoration: none; font-weight: bold; }
.link:hover { text-decoration: underline; }
</style>
</head>
<body>
<h1>🤖 Тестирование QR-кода и Telegram бота SmartSolTech</h1>
<div class="section instructions">
<h2>📋 Инструкции для тестирования</h2>
<p><strong>Что мы добавили:</strong></p>
<ul>
<li>✅ QR-код генерацию в модальном окне заявок</li>
<li>✅ Интеграция с Telegram ботом</li>
<li>✅ Автоматическое создание заявки через форму</li>
<li>✅ Подтверждение регистрации через Telegram</li>
</ul>
</div>
<div class="section test-case">
<h2>🔍 Тест-кейс 1: Создание заявки через модальное окно</h2>
<div class="step">
<h3>Шаг 1:</h3>
<p>Перейдите на страницу услуг: <a href="http://localhost:8000/services/" class="link" target="_blank">http://localhost:8000/services/</a></p>
</div>
<div class="step">
<h3>Шаг 2:</h3>
<p>Нажмите кнопку <strong>"Заказать услугу"</strong> под любой услугой</p>
</div>
<div class="step">
<h3>Шаг 3:</h3>
<p>Заполните форму в модальном окне:</p>
<ul>
<li>Имя: Тест</li>
<li>Фамилия: Пользователь</li>
<li>Email: test@example.com</li>
<li>Телефон: +7-123-456-7890</li>
<li>Описание: Тестовая заявка на услугу</li>
</ul>
</div>
<div class="step">
<h3>Шаг 4:</h3>
<p>Поставьте галочку согласия и нажмите <strong>"Отправить заявку"</strong></p>
</div>
<div class="step">
<h3>Ожидаемый результат:</h3>
<p>✅ Должен появиться QR-код для Telegram</p>
<p>✅ Кнопка "Открыть в Telegram" должна работать</p>
</div>
</div>
<div class="section test-case">
<h2>🤖 Тест-кейс 2: Подтверждение через Telegram бота</h2>
<div class="step">
<h3>Шаг 1:</h3>
<p>Отсканируйте QR-код телефоном или нажмите кнопку "Открыть в Telegram"</p>
</div>
<div class="step">
<h3>Шаг 2:</h3>
<p>Бот должен отправить команду /start с параметрами заявки</p>
</div>
<div class="step">
<h3>Шаг 3:</h3>
<p>Нажмите "Start" в Telegram</p>
</div>
<div class="step">
<h3>Ожидаемый результат:</h3>
<p>✅ Бот приветствует и подтверждает регистрацию</p>
<p>✅ Заявка получает статус "подтверждена"</p>
<p>✅ Клиент связывается с chat_id пользователя</p>
</div>
</div>
<div class="section warning">
<h2>⚠️ Возможные проблемы</h2>
<ul>
<li><strong>QR-код не генерируется:</strong> Проверьте настройки Telegram бота в админке</li>
<li><strong>Telegram бот не отвечает:</strong> Проверьте токен бота и что сервис telegram_bot запущен</li>
<li><strong>Ошибка 500:</strong> Проверьте логи Django в консоли Docker</li>
<li><strong>Ссылки не работают:</strong> Убедитесь, что JavaScript загружается корректно</li>
</ul>
</div>
<div class="section success">
<h2>📱 Быстрые ссылки для тестирования</h2>
<p><a href="http://localhost:8000/" class="link" target="_blank">🏠 Главная страница</a></p>
<p><a href="http://localhost:8000/services/" class="link" target="_blank">🛠️ Страница услуг (тестируем здесь)</a></p>
<p><a href="http://localhost:8000/about/" class="link" target="_blank"> О нас</a></p>
<p><a href="http://localhost:8080" class="link" target="_blank">🗄️ PgAdmin (admin:admin)</a></p>
</div>
<div class="section instructions">
<h2>🔧 Техническая информация</h2>
<h3>Новые компоненты:</h3>
<ul>
<li><strong>QR-код секция:</strong> Добавлена в модальное окно services_modern.html</li>
<li><strong>JavaScript:</strong> Обновлен для работы с generate_qr_code API</li>
<li><strong>Backend:</strong> generate_qr_code view создает QR-код и заявку</li>
<li><strong>Telegram интеграция:</strong> Обрабатывает /start команды с параметрами</li>
</ul>
<h3>Workflow:</h3>
<ol>
<li>Пользователь заполняет форму → JavaScript отправляет POST на /service/generate_qr_code/</li>
<li>Django создает Client, ServiceRequest и генерирует QR-код</li>
<li>Возвращается ссылка на Telegram бота с токеном</li>
<li>Пользователь сканирует QR или переходит по ссылке</li>
<li>Telegram бот обрабатывает /start команду и подтверждает заявку</li>
</ol>
</div>
</body>
</html>