first commit
This commit is contained in:
264
src/components/PartDetailCard.tsx
Normal file
264
src/components/PartDetailCard.tsx
Normal file
@ -0,0 +1,264 @@
|
||||
import React, { useState } from 'react';
|
||||
import { useRouter } from 'next/router';
|
||||
import { useLazyQuery } from '@apollo/client';
|
||||
import { LaximoOEMResult } from '@/types/laximo';
|
||||
import { SEARCH_LAXIMO_OEM } from '@/lib/graphql';
|
||||
import BrandSelectionModal from './BrandSelectionModal';
|
||||
|
||||
interface PartDetailCardProps {
|
||||
oem: string;
|
||||
name: string;
|
||||
brand?: string;
|
||||
description?: string;
|
||||
catalogCode: string;
|
||||
vehicleId: string;
|
||||
ssd: string;
|
||||
isExpanded?: boolean;
|
||||
onToggleExpand?: () => void;
|
||||
}
|
||||
|
||||
const PartDetailCard: React.FC<PartDetailCardProps> = ({
|
||||
oem,
|
||||
name,
|
||||
brand,
|
||||
description,
|
||||
catalogCode,
|
||||
vehicleId,
|
||||
ssd,
|
||||
isExpanded = false,
|
||||
onToggleExpand
|
||||
}) => {
|
||||
const router = useRouter();
|
||||
const [localExpanded, setLocalExpanded] = useState(false);
|
||||
const [isBrandModalOpen, setIsBrandModalOpen] = useState(false);
|
||||
|
||||
// Используем локальное состояние если нет внешнего контроля
|
||||
const expanded = onToggleExpand ? isExpanded : localExpanded;
|
||||
const toggleExpand = onToggleExpand || (() => setLocalExpanded(!localExpanded));
|
||||
|
||||
const [executeOEMSearch, { data, loading, error }] = useLazyQuery(SEARCH_LAXIMO_OEM, {
|
||||
errorPolicy: 'all'
|
||||
});
|
||||
|
||||
const handleToggleExpand = () => {
|
||||
toggleExpand();
|
||||
|
||||
// Загружаем данные только при первом раскрытии
|
||||
if (!expanded && !data && !loading) {
|
||||
executeOEMSearch({
|
||||
variables: { catalogCode, vehicleId, oemNumber: oem, ssd }
|
||||
});
|
||||
}
|
||||
};
|
||||
|
||||
const handleFindOffers = () => {
|
||||
console.log('🔍 Выбрана деталь для поиска предложений:', name, 'OEM:', oem);
|
||||
// Показываем модал выбора бренда
|
||||
setIsBrandModalOpen(true);
|
||||
};
|
||||
|
||||
const handleCloseBrandModal = () => {
|
||||
setIsBrandModalOpen(false);
|
||||
};
|
||||
|
||||
const handleOpenFullInfo = () => {
|
||||
// Переход на отдельную страницу с детальной информацией о детали
|
||||
const url = `/vehicle-search/${catalogCode}/${vehicleId}/part/${oem}?use_storage=1&ssd_length=${ssd.length}`;
|
||||
router.push(url);
|
||||
};
|
||||
|
||||
const oemResult: LaximoOEMResult | null = data?.laximoOEMSearch || null;
|
||||
|
||||
const totalUnits = oemResult?.categories.reduce((total, cat) => total + cat.units.length, 0) || 0;
|
||||
|
||||
return (
|
||||
<div className="bg-white border border-gray-200 rounded-lg overflow-hidden hover:shadow-md hover:border-red-300 transition-all duration-200 cursor-pointer">
|
||||
{/* Основная информация - кликабельная область */}
|
||||
<div
|
||||
className="p-4 hover:bg-gray-50 transition-colors"
|
||||
onClick={handleFindOffers}
|
||||
>
|
||||
<div className="flex items-start justify-between">
|
||||
<div className="flex-1">
|
||||
<h3 className="text-lg font-semibold text-gray-900 mb-2 hover:text-red-600 transition-colors">
|
||||
{name}
|
||||
</h3>
|
||||
|
||||
<div className="flex flex-wrap items-center gap-4 mb-3">
|
||||
<div className="flex items-center space-x-2">
|
||||
<span className="text-sm text-gray-500">OEM:</span>
|
||||
<span className="font-mono text-sm font-medium text-red-600 bg-red-50 px-2 py-1 rounded">
|
||||
{oem}
|
||||
</span>
|
||||
</div>
|
||||
|
||||
{brand && (
|
||||
<div className="flex items-center space-x-2">
|
||||
<span className="text-sm text-gray-500">Бренд:</span>
|
||||
<span className="text-sm font-medium text-blue-600">{brand}</span>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{description && (
|
||||
<p className="text-sm text-gray-600 mb-3">{description}</p>
|
||||
)}
|
||||
|
||||
{/* Подсказка о переходе */}
|
||||
<div className="text-sm text-gray-500 italic">
|
||||
Нажмите, чтобы найти предложения для этой детали
|
||||
</div>
|
||||
|
||||
{/* Краткая информация о применимости */}
|
||||
{oemResult && (
|
||||
<div className="text-sm text-gray-600 mt-2">
|
||||
<span className="font-medium">Применимость:</span>
|
||||
<span className="ml-1">
|
||||
{oemResult.categories.length} категорий, {totalUnits} узлов
|
||||
</span>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{/* Иконка перехода */}
|
||||
<div className="ml-4 text-gray-400">
|
||||
<svg className="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M9 5l7 7-7 7" />
|
||||
</svg>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Дополнительные действия */}
|
||||
<div className="px-4 pb-4 flex flex-wrap gap-2 border-t border-gray-100">
|
||||
<button
|
||||
onClick={(e) => {
|
||||
e.stopPropagation();
|
||||
handleToggleExpand();
|
||||
}}
|
||||
className="inline-flex items-center px-3 py-2 border border-gray-300 text-sm font-medium rounded-md text-gray-700 bg-white hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-red-500 focus:border-red-500 transition-colors"
|
||||
>
|
||||
<svg
|
||||
className={`w-4 h-4 mr-2 transform transition-transform duration-200 ${expanded ? 'rotate-180' : ''}`}
|
||||
fill="none"
|
||||
stroke="currentColor"
|
||||
viewBox="0 0 24 24"
|
||||
>
|
||||
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M19 9l-7 7-7-7" />
|
||||
</svg>
|
||||
{expanded ? 'Скрыть применимость' : 'Показать применимость'}
|
||||
</button>
|
||||
|
||||
<button
|
||||
onClick={(e) => {
|
||||
e.stopPropagation();
|
||||
handleOpenFullInfo();
|
||||
}}
|
||||
className="inline-flex items-center px-3 py-2 border border-gray-300 text-sm font-medium rounded-md text-gray-700 bg-white hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-red-500 focus:border-red-500 transition-colors"
|
||||
>
|
||||
<svg className="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z" />
|
||||
</svg>
|
||||
Подробно
|
||||
</button>
|
||||
</div>
|
||||
|
||||
{/* Развернутая информация */}
|
||||
{expanded && (
|
||||
<div className="border-t border-gray-200 bg-gray-50">
|
||||
<div className="p-4">
|
||||
<h4 className="text-sm font-medium text-gray-900 mb-3">
|
||||
Применимость в автомобиле:
|
||||
</h4>
|
||||
|
||||
{loading && (
|
||||
<div className="flex items-center justify-center py-6">
|
||||
<div className="animate-spin rounded-full h-6 w-6 border-b-2 border-red-600"></div>
|
||||
<span className="ml-2 text-sm text-gray-600">Загружаем информацию...</span>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{error && (
|
||||
<div className="text-sm text-red-600 bg-red-50 border border-red-200 rounded-lg p-3">
|
||||
Ошибка загрузки: {error.message}
|
||||
</div>
|
||||
)}
|
||||
|
||||
{oemResult && (
|
||||
<div className="space-y-3">
|
||||
{oemResult.categories.map((category) => (
|
||||
<div key={category.categoryid} className="bg-white rounded-lg border border-gray-200 p-3">
|
||||
<h5 className="text-sm font-semibold text-gray-900 mb-2 flex items-center">
|
||||
<svg className="w-4 h-4 mr-2 text-gray-500" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M19 11H5m14 0a2 2 0 012 2v6a2 2 0 01-2 2H5a2 2 0 01-2-2v-6a2 2 0 012-2m14 0V9a2 2 0 00-2-2M5 11V9a2 2 0 012-2m0 0V5a2 2 0 012-2h6a2 2 0 012 2v2M7 7h10" />
|
||||
</svg>
|
||||
{category.name}
|
||||
</h5>
|
||||
|
||||
<div className="space-y-2">
|
||||
{category.units.map((unit) => (
|
||||
<div key={unit.unitid} className="ml-4 border-l-2 border-gray-200 pl-3">
|
||||
<div className="text-sm font-medium text-gray-800 flex items-center">
|
||||
<svg className="w-3 h-3 mr-2 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M10.325 4.317c.426-1.756 2.924-1.756 3.35 0a1.724 1.724 0 002.573 1.066c1.543-.94 3.31.826 2.37 2.37a1.724 1.724 0 001.065 2.572c1.756.426 1.756 2.924 0 3.35a1.724 1.724 0 00-1.066 2.573c.94 1.543-.826 3.31-2.37 2.37a1.724 1.724 0 00-2.572 1.065c-.426 1.756-2.924 1.756-3.35 0a1.724 1.724 0 00-2.573-1.066c-1.543.94-3.31-.826-2.37-2.37a1.724 1.724 0 00-1.065-2.572c-1.756-.426-1.756-2.924 0-3.35a1.724 1.724 0 001.066-2.573c-.94-1.543.826-3.31 2.37-2.37.996.608 2.296.07 2.572-1.065z" />
|
||||
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M15 12a3 3 0 11-6 0 3 3 0 016 0z" />
|
||||
</svg>
|
||||
{unit.name}
|
||||
{unit.code && (
|
||||
<span className="ml-2 text-xs text-gray-500">({unit.code})</span>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{unit.details.map((detail, index) => (
|
||||
<div key={`${detail.detailid}-${index}`} className="ml-4 mt-1">
|
||||
<div className="text-xs text-gray-600 flex items-start">
|
||||
<svg className="w-3 h-3 mr-1 mt-0.5 flex-shrink-0 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z" />
|
||||
</svg>
|
||||
<div className="flex-1">
|
||||
<div>{detail.name}</div>
|
||||
{detail.amount && (
|
||||
<div className="mt-1">
|
||||
<span className="bg-blue-100 text-blue-800 px-2 py-0.5 rounded text-xs">
|
||||
Количество: {detail.amount}
|
||||
</span>
|
||||
</div>
|
||||
)}
|
||||
{detail.range && (
|
||||
<div className="mt-1 text-xs text-gray-500">
|
||||
Период: {detail.range}
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
)}
|
||||
|
||||
{!loading && !error && !oemResult && (
|
||||
<div className="text-sm text-gray-500 text-center py-4">
|
||||
Информация о применимости не найдена
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{/* Модал выбора бренда */}
|
||||
<BrandSelectionModal
|
||||
isOpen={isBrandModalOpen}
|
||||
onClose={handleCloseBrandModal}
|
||||
articleNumber={oem}
|
||||
detailName={name}
|
||||
/>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
export default PartDetailCard;
|
||||
Reference in New Issue
Block a user