mirror of
https://github.com/ManInDark/HabitTrove.git
synced 2026-01-20 22:24:28 +01:00
Added notification for admin user (#106)
This commit is contained in:
135
components/NotificationDropdown.tsx
Normal file
135
components/NotificationDropdown.tsx
Normal file
@@ -0,0 +1,135 @@
|
||||
import React from 'react';
|
||||
import { ScrollArea } from '@/components/ui/scroll-area';
|
||||
import { Separator } from '@/components/ui/separator';
|
||||
import { Avatar, AvatarFallback, AvatarImage } from '@/components/ui/avatar';
|
||||
import { CoinsData, HabitsData, WishlistData, UserData, User, CoinTransaction } from '@/lib/types';
|
||||
import { t2d } from '@/lib/utils';
|
||||
import Link from 'next/link';
|
||||
import { DropdownMenuItem } from '@/components/ui/dropdown-menu';
|
||||
import { Info } from 'lucide-react';
|
||||
import {
|
||||
Tooltip,
|
||||
TooltipContent,
|
||||
TooltipProvider,
|
||||
TooltipTrigger,
|
||||
} from '@/components/ui/tooltip';
|
||||
|
||||
interface NotificationDropdownProps {
|
||||
currentUser: User | null;
|
||||
unreadNotifications: CoinTransaction[];
|
||||
displayedReadNotifications: CoinTransaction[];
|
||||
habitsData: HabitsData; // Keep needed props
|
||||
wishlistData: WishlistData;
|
||||
usersData: UserData;
|
||||
}
|
||||
|
||||
// Helper function to generate notification message
|
||||
const getNotificationMessage = (tx: CoinTransaction, triggeringUser?: User, relatedItemName?: string): string => {
|
||||
const username = triggeringUser?.username || 'Someone';
|
||||
const itemName = relatedItemName || 'a shared item';
|
||||
switch (tx.type) {
|
||||
case 'HABIT_COMPLETION':
|
||||
case 'TASK_COMPLETION':
|
||||
return `${username} completed ${itemName}.`;
|
||||
case 'WISH_REDEMPTION':
|
||||
return `${username} redeemed ${itemName}.`;
|
||||
// Add other relevant transaction types if needed
|
||||
default:
|
||||
return `Activity related to ${itemName} by ${username}.`; // Fallback message
|
||||
}
|
||||
};
|
||||
|
||||
// Helper function to get the name of the related item
|
||||
const getRelatedItemName = (tx: CoinTransaction, habitsData: HabitsData, wishlistData: WishlistData): string | undefined => {
|
||||
if (!tx.relatedItemId) return undefined;
|
||||
if (tx.type === 'HABIT_COMPLETION' || tx.type === 'TASK_COMPLETION') {
|
||||
return habitsData.habits.find(h => h.id === tx.relatedItemId)?.name;
|
||||
}
|
||||
if (tx.type === 'WISH_REDEMPTION') {
|
||||
return wishlistData.items.find(w => w.id === tx.relatedItemId)?.name;
|
||||
}
|
||||
return undefined;
|
||||
};
|
||||
|
||||
|
||||
export default function NotificationDropdown({
|
||||
currentUser,
|
||||
unreadNotifications, // Use props directly
|
||||
displayedReadNotifications, // Use props directly
|
||||
habitsData,
|
||||
wishlistData,
|
||||
usersData,
|
||||
}: NotificationDropdownProps) {
|
||||
if (!currentUser) {
|
||||
return <div className="p-4 text-sm text-gray-500">Not logged in.</div>;
|
||||
}
|
||||
|
||||
// Removed the useMemo block for calculating notifications
|
||||
|
||||
const renderNotification = (tx: CoinTransaction, isUnread: boolean) => {
|
||||
const triggeringUser = usersData.users.find(u => u.id === tx.userId);
|
||||
const relatedItemName = getRelatedItemName(tx, habitsData, wishlistData);
|
||||
const message = getNotificationMessage(tx, triggeringUser, relatedItemName);
|
||||
const txTimestamp = t2d({ timestamp: tx.timestamp, timezone: 'UTC' });
|
||||
const timeAgo = txTimestamp.toRelative(); // e.g., "2 hours ago"
|
||||
// Add the triggering user's ID to the query params if it exists
|
||||
const linkHref = `/coins?highlight=${tx.id}${tx.userId ? `&user=${tx.userId}` : ''}`;
|
||||
|
||||
return (
|
||||
// Wrap the Link with DropdownMenuItem and use asChild to pass props
|
||||
<DropdownMenuItem key={tx.id} asChild className={`p-0 focus:bg-inherit dark:focus:bg-inherit cursor-pointer`}>
|
||||
<Link href={linkHref} className={`block hover:bg-gray-100 dark:hover:bg-gray-800 transition-colors ${isUnread ? 'bg-blue-50 dark:bg-blue-900/20' : ''}`} scroll={true}>
|
||||
<div className="p-3 flex items-start gap-3">
|
||||
<Avatar className="h-8 w-8 mt-1">
|
||||
<AvatarImage src={triggeringUser?.avatarPath ? `/api/avatars/${triggeringUser.avatarPath.split('/').pop()}` : undefined} alt={triggeringUser?.username} />
|
||||
<AvatarFallback>{triggeringUser?.username?.charAt(0).toUpperCase() || '?'}</AvatarFallback>
|
||||
</Avatar>
|
||||
<div className="flex-1">
|
||||
<p className={`text-sm ${isUnread ? 'font-semibold' : ''}`}>{message}</p>
|
||||
<p className="text-xs text-gray-500 dark:text-gray-400">{timeAgo}</p>
|
||||
</div>
|
||||
</div>
|
||||
</Link>
|
||||
</DropdownMenuItem>
|
||||
);
|
||||
};
|
||||
|
||||
return (
|
||||
<TooltipProvider>
|
||||
{/* Removed the outer div as width is now set on DropdownMenuContent in NotificationBell */}
|
||||
<>
|
||||
<div className="p-3 border-b border-gray-200 dark:border-gray-700 flex items-center gap-2">
|
||||
<h4 className="text-sm font-medium">Notifications</h4>
|
||||
<Tooltip>
|
||||
<TooltipTrigger asChild>
|
||||
<Info className="h-4 w-4 text-muted-foreground cursor-help" />
|
||||
</TooltipTrigger>
|
||||
<TooltipContent side="left" className="max-w-xs">
|
||||
<p className="text-xs">
|
||||
Shows completions or redemptions by other users for habits or wishlist that you shared with them (you must be admin)
|
||||
</p>
|
||||
</TooltipContent>
|
||||
</Tooltip>
|
||||
</div>
|
||||
<ScrollArea className="h-[400px]">
|
||||
{unreadNotifications.length === 0 && displayedReadNotifications.length === 0 && (
|
||||
<div className="p-4 text-center text-sm text-gray-500">No notifications yet.</div>
|
||||
)}
|
||||
|
||||
{unreadNotifications.length > 0 && (
|
||||
<>
|
||||
{unreadNotifications.map(tx => renderNotification(tx, true))}
|
||||
{displayedReadNotifications.length > 0 && <Separator className="my-2" />}
|
||||
</>
|
||||
)}
|
||||
|
||||
{displayedReadNotifications.length > 0 && (
|
||||
<>
|
||||
{displayedReadNotifications.map(tx => renderNotification(tx, false))}
|
||||
</>
|
||||
)}
|
||||
</ScrollArea>
|
||||
</> {/* Close the fragment */}
|
||||
</TooltipProvider>
|
||||
);
|
||||
}
|
||||
Reference in New Issue
Block a user