first commit

This commit is contained in:
Bivekich
2025-06-26 06:59:59 +03:00
commit d44874775c
450 changed files with 76635 additions and 0 deletions

View File

@ -0,0 +1,126 @@
import React from "react";
import CatalogProductCard from "./CatalogProductCard";
import { useArticleImage } from "@/hooks/useArticleImage";
import { useCart } from "@/contexts/CartContext";
import { toast } from "react-hot-toast";
interface CartRecommendedProps {
recommendedProducts?: any[];
isLoadingPrices?: boolean;
}
// Компонент для отдельной карточки рекомендуемого товара с реальным изображением
const RecommendedProductCard: React.FC<{
item: any;
isLoadingPrice: boolean;
formatPrice: (price: number | null, isLoading: boolean) => string;
}> = ({ item, isLoadingPrice, formatPrice }) => {
const { imageUrl } = useArticleImage(item.artId, { enabled: !!item.artId });
const { addItem } = useCart();
// Если нет изображения, используем заглушку с иконкой (но не мокап-фотку)
const displayImage = imageUrl || '';
// Обработчик добавления в корзину с тоастером
const handleAddToCart = async (e: React.MouseEvent) => {
e.preventDefault();
e.stopPropagation();
try {
// Извлекаем цену как число
const numericPrice = item.minPrice || 0;
if (numericPrice <= 0) {
toast.error('Цена товара не найдена');
return;
}
// Добавляем товар в корзину
addItem({
productId: String(item.artId) || undefined,
name: item.name || `${item.brand} ${item.articleNumber}`,
description: item.name || `${item.brand} ${item.articleNumber}`,
price: numericPrice,
currency: 'RUB',
quantity: 1,
image: displayImage,
brand: item.brand,
article: item.articleNumber,
supplier: 'AutoEuro',
deliveryTime: '1 день',
isExternal: true
});
// Показываем успешный тоастер
toast.success(
<div>
<div className="font-semibold">Товар добавлен в корзину!</div>
<div className="text-sm text-gray-600">{item.name || `${item.brand} ${item.articleNumber}`}</div>
</div>,
{
duration: 3000,
icon: '🛒',
}
);
} catch (error) {
console.error('Ошибка добавления в корзину:', error);
toast.error('Ошибка при добавлении товара в корзину');
}
};
return (
<CatalogProductCard
image={displayImage}
discount=""
price={formatPrice(item.minPrice, isLoadingPrice && item.minPrice === undefined)}
oldPrice=""
title={item.name || `${item.brand} ${item.articleNumber}`}
brand={item.brand}
articleNumber={item.articleNumber}
brandName={item.brand}
artId={item.artId}
productId={item.artId ? String(item.artId) : undefined} // Добавляем productId для работы избранного
currency="RUB"
onAddToCart={handleAddToCart} // Передаем обработчик добавления в корзину
/>
);
};
const CartRecommended: React.FC<CartRecommendedProps> = ({
recommendedProducts = [],
isLoadingPrices = false
}) => {
// Фильтруем и ограничиваем количество рекомендаций
const validRecommendations = recommendedProducts
.filter(item => item && item.brand && item.articleNumber) // Фильтруем только валидные товары
.slice(0, 5); // Ограничиваем до 5 товаров
// Если нет валидных рекомендаций, не показываем блок
if (validRecommendations.length === 0) {
return null;
}
const formatPrice = (price: number | null, isLoading: boolean = false) => {
if (isLoading) return "Загрузка...";
if (!price) return "По запросу";
return `от ${price.toLocaleString('ru-RU')}`;
};
return (
<>
<h2 className="heading-11">Рекомендуемые</h2>
<div className="w-layout-hflex core-product-search">
{validRecommendations.map((item, idx) => (
<RecommendedProductCard
key={`${item.brand}-${item.articleNumber}-${idx}`}
item={item}
isLoadingPrice={isLoadingPrices}
formatPrice={formatPrice}
/>
))}
</div>
</>
);
};
export default CartRecommended;