Новые измнения в коде

This commit is contained in:
Web-serfer 2026-04-19 20:19:24 +05:00
parent beeec4740e
commit 4b9735118b
4 changed files with 251 additions and 21 deletions

View file

@ -0,0 +1,113 @@
/// <reference path="../pb_data/types.d.ts" />
migrate((app) => {
const collection = new Collection({
"createRule": null,
"deleteRule": null,
"fields": [
{
"autogeneratePattern": "[a-z0-9]{15}",
"hidden": false,
"id": "text3208210256",
"max": 15,
"min": 15,
"name": "id",
"pattern": "^[a-z0-9]+$",
"presentable": false,
"primaryKey": true,
"required": true,
"system": true,
"type": "text"
},
{
"autogeneratePattern": "",
"hidden": false,
"id": "text1579384326",
"max": 0,
"min": 0,
"name": "name",
"pattern": "",
"presentable": false,
"primaryKey": false,
"required": false,
"system": false,
"type": "text"
},
{
"autogeneratePattern": "",
"hidden": false,
"id": "text1146066909",
"max": 0,
"min": 0,
"name": "phone",
"pattern": "",
"presentable": false,
"primaryKey": false,
"required": false,
"system": false,
"type": "text"
},
{
"autogeneratePattern": "",
"hidden": false,
"id": "text3785202386",
"max": 0,
"min": 0,
"name": "service",
"pattern": "",
"presentable": false,
"primaryKey": false,
"required": false,
"system": false,
"type": "text"
},
{
"autogeneratePattern": "",
"hidden": false,
"id": "text2063623452",
"max": 0,
"min": 0,
"name": "status",
"pattern": "",
"presentable": false,
"primaryKey": false,
"required": false,
"system": false,
"type": "text"
},
{
"hidden": false,
"id": "autodate2990389176",
"name": "created",
"onCreate": true,
"onUpdate": false,
"presentable": false,
"system": false,
"type": "autodate"
},
{
"hidden": false,
"id": "autodate3332085495",
"name": "updated",
"onCreate": true,
"onUpdate": true,
"presentable": false,
"system": false,
"type": "autodate"
}
],
"id": "pbc_3441013282",
"indexes": [],
"listRule": null,
"name": "consultations",
"system": false,
"type": "base",
"updateRule": null,
"viewRule": null
});
return app.save(collection);
}, (app) => {
const collection = app.findCollectionByNameOrId("pbc_3441013282");
return app.delete(collection);
})

View file

@ -0,0 +1,24 @@
/// <reference path="../pb_data/types.d.ts" />
migrate((app) => {
const collection = app.findCollectionByNameOrId("pbc_3441013282")
// update collection data
unmarshal({
"createRule": "",
"deleteRule": "@request.auth.id != \"\"",
"updateRule": "@request.auth.id != \"\""
}, collection)
return app.save(collection)
}, (app) => {
const collection = app.findCollectionByNameOrId("pbc_3441013282")
// update collection data
unmarshal({
"createRule": null,
"deleteRule": null,
"updateRule": null
}, collection)
return app.save(collection)
})

View file

@ -69,6 +69,7 @@ const title = 'Бесплатная консультация';
modal.classList.add('active'); modal.classList.add('active');
modal.setAttribute('aria-hidden', 'false'); modal.setAttribute('aria-hidden', 'false');
document.body.style.overflow = 'hidden'; document.body.style.overflow = 'hidden';
startAutoCloseTimer();
} }
function closeModal() { function closeModal() {
@ -76,6 +77,24 @@ const title = 'Бесплатная консультация';
modal.classList.remove('active'); modal.classList.remove('active');
modal.setAttribute('aria-hidden', 'true'); modal.setAttribute('aria-hidden', 'true');
document.body.style.overflow = ''; document.body.style.overflow = '';
resetAutoCloseTimer();
}
// Таймер для автозакрытия
let autoCloseTimer: ReturnType<typeof setTimeout> | null = null;
function startAutoCloseTimer() {
if (autoCloseTimer) clearTimeout(autoCloseTimer);
autoCloseTimer = setTimeout(() => {
closeModal();
}, 5000);
}
function resetAutoCloseTimer() {
if (autoCloseTimer) {
clearTimeout(autoCloseTimer);
autoCloseTimer = null;
}
} }
// Открытие по кастомному событию // Открытие по кастомному событию
@ -119,7 +138,7 @@ const title = 'Бесплатная консультация';
const phone = formData.get('phone'); const phone = formData.get('phone');
if (!name || !phone) { if (!name || !phone) {
alert('Пожалуйста, заполните имя и телефон'); (window as any).toast?.show('Пожалуйста, заполните имя и телефон', 'error', 3000);
return; return;
} }
@ -133,18 +152,22 @@ const title = 'Бесплатная консультация';
const result = await response.json(); const result = await response.json();
if (result.success) { if (result.success) {
alert('Спасибо! Мы свяжемся с вами в течение 15 минут.'); (window as any).toast?.show('Спасибо! Мы свяжемся с вами в течение 15 минут.', 'success', 4000);
form.reset(); form.reset();
closeModal(); closeModal();
} else { } else {
alert('Ошибка: ' + result.error); (window as any).toast?.show(result.error || 'Ошибка отправки', 'error', 4000);
} }
} catch (error) { } catch (error) {
console.error('Error:', error); console.error('Error:', error);
alert('Ошибка отправки. Попробуйте позже.'); (window as any).toast?.show('Ошибка отправки. Попробуйте позже.', 'error', 4000);
} }
}); });
// Сброс таймера при активности пользователя
modal?.addEventListener('mousemove', resetAutoCloseTimer);
modal?.addEventListener('click', resetAutoCloseTimer);
// Для Astro View Transitions // Для Astro View Transitions
document.addEventListener('astro:page-load', () => { document.addEventListener('astro:page-load', () => {
// Переинициализация при навигации // Переинициализация при навигации

View file

@ -1,5 +1,17 @@
import type { APIRoute } from 'astro'; import type { APIRoute } from 'astro';
import { pb } from '../../lib/pb'; import nodemailer from 'nodemailer';
const POCKETBASE_URL = import.meta.env.POCKETBASE_URL || 'http://localhost:8090';
const SMTP_HOST = import.meta.env.SMTP_HOST || 'localhost';
const SMTP_PORT = import.meta.env.SMTP_PORT || '1025';
const NOTIFY_EMAIL = import.meta.env.NOTIFY_EMAIL || 'info@avtourist.ru';
const transporter = nodemailer.createTransport({
host: SMTP_HOST,
port: parseInt(SMTP_PORT),
secure: false,
ignoreTLS: true,
});
const RATE_LIMIT_WINDOW = 60 * 1000; const RATE_LIMIT_WINDOW = 60 * 1000;
const MAX_REQUESTS = 3; const MAX_REQUESTS = 3;
@ -37,55 +49,113 @@ export const POST: APIRoute = async ({ request }) => {
return new Response(JSON.stringify({ return new Response(JSON.stringify({
success: false, success: false,
error: 'Слишком много запросов. Попробуйте позже.' error: 'Слишком много запросов. Попробуйте позже.'
}), { status: 429 }); }), { status: 429, headers: { 'Content-Type': 'application/json' } });
} }
try { try {
const data = await request.json(); const data = await request.json();
const { name, phone, service, website } = data; const { name, phone, website } = data;
if (website) { if (website) {
return new Response(JSON.stringify({ return new Response(JSON.stringify({
success: false, success: false,
error: 'Спам обнаружен' error: 'Спам обнаружен'
}), { status: 400 }); }), { status: 400, headers: { 'Content-Type': 'application/json' } });
} }
if (!name || !phone) { if (!name || !phone) {
return new Response(JSON.stringify({ return new Response(JSON.stringify({
success: false, success: false,
error: 'Имя и телефон обязательны' error: 'Имя и телефон обязательны'
}), { status: 400 }); }), { status: 400, headers: { 'Content-Type': 'application/json' } });
} }
if (name.length < 2 || name.length > 100) { if (name.length < 2 || name.length > 100) {
return new Response(JSON.stringify({ return new Response(JSON.stringify({
success: false, success: false,
error: 'Некорректное имя' error: 'Некорректное имя'
}), { status: 400 }); }), { status: 400, headers: { 'Content-Type': 'application/json' } });
} }
if (!validatePhone(phone)) { if (!validatePhone(phone)) {
return new Response(JSON.stringify({ return new Response(JSON.stringify({
success: false, success: false,
error: 'Некорректный номер телефона' error: 'Некорректный номер телефона'
}), { status: 400 }); }), { status: 400, headers: { 'Content-Type': 'application/json' } });
} }
const record = await pb.collection('consultations').create({ const cleanPhone = phone.replace(/\D/g, '');
const response = await fetch(
`${POCKETBASE_URL}/api/collections/consultations/records`,
{
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({
name: name.trim(), name: name.trim(),
phone: phone.replace(/\D/g, ''), phone: cleanPhone,
service: service || '',
status: 'new', status: 'new',
created_at: new Date().toISOString(), }),
}
);
const result = await response.json();
if (!response.ok) {
return new Response(JSON.stringify({
success: false,
error: result.message || 'Ошибка при отправке заявки'
}), { status: response.status, headers: { 'Content-Type': 'application/json' } });
}
const emailHtml = `
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
</head>
<body style="margin: 0; padding: 20px; font-family: Arial, sans-serif; background-color: #f5f7fa;">
<div style="max-width: 600px; margin: 0 auto; background: #ffffff; border-radius: 12px; padding: 30px;">
<h2 style="color: #1e3050; margin: 0 0 20px;">Новая заявка на консультацию!</h2>
<table style="width: 100%; border-collapse: collapse;">
<tr>
<td style="padding: 10px 0; border-bottom: 1px solid #e0e4e8;"><strong>Имя:</strong></td>
<td style="padding: 10px 0; border-bottom: 1px solid #e0e4e8;">${name}</td>
</tr>
<tr>
<td style="padding: 10px 0; border-bottom: 1px solid #e0e4e8;"><strong>Телефон:</strong></td>
<td style="padding: 10px 0; border-bottom: 1px solid #e0e4e8;"><a href="tel:${cleanPhone}">${phone}</a></td>
</tr>
<tr>
<td style="padding: 10px 0;"><strong>Дата:</strong></td>
<td style="padding: 10px 0;">${new Date().toLocaleString('ru-RU')}</td>
</tr>
</table>
</div>
</body>
</html>
`.trim();
try {
await transporter.sendMail({
from: 'avtourist@surgut.ru',
to: NOTIFY_EMAIL,
subject: `Новая заявка от ${name}`,
html: emailHtml,
}); });
console.log('Email notification sent');
} catch (emailError) {
console.error('Email send error:', emailError);
}
return new Response(JSON.stringify({ return new Response(JSON.stringify({
success: true, success: true,
message: 'Заявка отправлена! Мы свяжемся с вами в течение 15 минут.', message: 'Заявка отправлена! Мы свяжемся с вами в течение 15 минут.',
id: record.id id: result.id
}), { status: 201 }); }), { status: 201, headers: { 'Content-Type': 'application/json' } });
} catch (error: any) { } catch (error: any) {
console.error('Consultation error:', error); console.error('Consultation error:', error);
@ -93,6 +163,6 @@ export const POST: APIRoute = async ({ request }) => {
return new Response(JSON.stringify({ return new Response(JSON.stringify({
success: false, success: false,
error: error.message || 'Ошибка при отправке заявки' error: error.message || 'Ошибка при отправке заявки'
}), { status: 400 }); }), { status: 400, headers: { 'Content-Type': 'application/json' } });
} }
}; };