Some checks failed
Deploy to Web Server flyer-crawler.projectium.com / deploy (push) Failing after 1m5s
98 lines
4.5 KiB
TypeScript
98 lines
4.5 KiB
TypeScript
// src/features/charts/PriceChart.tsx
|
|
import React from 'react';
|
|
import type { DealItem, User } from '../../types';
|
|
import { TagIcon } from '../../components/icons/TagIcon';
|
|
import { LoadingSpinner } from '../../components/LoadingSpinner';
|
|
import { formatUnitPrice } from '../../utils/unitConverter';
|
|
import { UserIcon } from '../../components/icons/UserIcon';
|
|
|
|
interface PriceChartProps {
|
|
deals: DealItem[];
|
|
isLoading: boolean;
|
|
unitSystem: 'metric' | 'imperial';
|
|
user: User | null;
|
|
}
|
|
|
|
export const PriceChart: React.FC<PriceChartProps> = ({ deals, isLoading, unitSystem, user }) => {
|
|
const renderContent = () => {
|
|
if (!user) {
|
|
return (
|
|
<div className="flex flex-col items-center justify-center h-full min-h-[150px] text-center">
|
|
<UserIcon className="w-10 h-10 text-gray-400 mb-3" />
|
|
<h4 className="font-semibold text-gray-700 dark:text-gray-300">Personalized Deals</h4>
|
|
<p className="text-sm text-gray-500 dark:text-gray-400 mt-1">
|
|
Log in to see active deals for items on your watchlist.
|
|
</p>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
if (isLoading) {
|
|
return (
|
|
<div role="status" className="flex justify-center items-center h-full min-h-[100px]">
|
|
<div className="w-6 h-6 text-brand-primary"><LoadingSpinner /></div> <span className="ml-2 text-sm text-gray-500 dark:text-gray-400">Finding active deals...</span>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
if (deals.length === 0) {
|
|
return <p className="text-sm text-gray-500 dark:text-gray-400 text-center py-4">No deals for your watched items found in any currently valid flyers.</p>;
|
|
}
|
|
|
|
return (
|
|
<div className="overflow-y-auto max-h-80">
|
|
<table className="min-w-full text-sm">
|
|
<thead className="bg-gray-50 dark:bg-gray-800 sticky top-0 z-10">
|
|
<tr>
|
|
<th className="px-4 py-2 text-left font-medium text-gray-600 dark:text-gray-300">Item</th>
|
|
<th className="px-4 py-2 text-left font-medium text-gray-600 dark:text-gray-300">Store</th>
|
|
<th className="px-4 py-2 text-right font-medium text-gray-600 dark:text-gray-300">Price</th>
|
|
<th className="px-4 py-2 text-right font-medium text-gray-600 dark:text-gray-300">Unit Price</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody className="divide-y divide-gray-200 dark:divide-gray-700">
|
|
{deals.map((deal, index) => {
|
|
// The formatUnitPrice function returns an object { price: string, unit: string }.
|
|
// We need to combine these into a single string for rendering and to match the test expectation.
|
|
const unitPriceData = deal.unit_price ? formatUnitPrice(deal.unit_price, unitSystem) : null;
|
|
const formattedUnitPriceString = unitPriceData
|
|
? `${unitPriceData.price}${unitPriceData.unit}`
|
|
: 'N/A';
|
|
|
|
return (
|
|
<tr key={`${deal.item}-${deal.storeName}-${index}`} className="hover:bg-gray-50 dark:hover:bg-gray-800/50">
|
|
<td className="px-4 py-2 font-semibold text-gray-900 dark:text-white">
|
|
<div className="flex justify-between items-baseline">
|
|
<span>{deal.item}</span>
|
|
{deal.master_item_name && deal.master_item_name.toLowerCase() !== deal.item.toLowerCase() && (
|
|
<span className="ml-2 text-xs font-normal italic text-gray-500 dark:text-gray-400 whitespace-nowrap">
|
|
({deal.master_item_name})
|
|
</span>
|
|
)}
|
|
</div>
|
|
<div className="text-xs text-gray-500 dark:text-gray-400 font-normal">{deal.quantity}</div>
|
|
</td>
|
|
<td className="px-4 py-2 text-left text-gray-700 dark:text-gray-200">{deal.storeName}</td>
|
|
<td className="px-4 py-2 text-right text-gray-700 dark:text-gray-200">{deal.price_display}</td>
|
|
<td className="px-4 py-2 text-right">
|
|
{formattedUnitPriceString}
|
|
</td>
|
|
</tr>
|
|
)
|
|
})}
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
return (
|
|
<div className="bg-white dark:bg-gray-900 rounded-lg border border-gray-200 dark:border-gray-700 p-4">
|
|
<h3 className="text-lg font-semibold mb-4 text-gray-800 dark:text-white flex items-center">
|
|
<TagIcon className="w-5 h-5 mr-2 text-brand-primary" />
|
|
Active Deals on Watched Items
|
|
</h3>
|
|
{renderContent()}
|
|
</div>
|
|
);
|
|
}; |