Files
tuxmate-abusoww/src/components/app/CategorySection.tsx

194 lines
7.0 KiB
TypeScript

'use client';
import { memo, useRef, useLayoutEffect, useEffect } from 'react';
import gsap from 'gsap';
import { type DistroId, type AppData, type Category } from '@/lib/data';
import { analytics } from '@/lib/analytics';
import { CategoryHeader } from './CategoryHeader';
import { AppItem } from './AppItem';
// Category section.
interface CategorySectionProps {
category: Category;
categoryApps: AppData[];
selectedApps: Set<string>;
isAppAvailable: (id: string) => boolean;
selectedDistro: DistroId;
toggleApp: (id: string) => void;
isExpanded: boolean;
onToggleExpanded: () => void;
focusedId: string | undefined;
focusedType: 'category' | 'app' | undefined;
onTooltipEnter: (t: string, e: React.MouseEvent) => void;
onTooltipLeave: () => void;
categoryIndex: number;
onCategoryFocus?: () => void;
onAppFocus?: (appId: string) => void;
isVerified?: (distro: DistroId, packageName: string) => boolean;
getVerificationSource?: (distro: DistroId, packageName: string) => 'flathub' | 'snap' | null;
}
const categoryColors: Record<Category, string> = {
'Web Browsers': 'orange',
'Communication': 'blue',
'Media': 'yellow',
'Gaming': 'purple',
'Office': 'indigo',
'Creative': 'cyan',
'System': 'red',
'File Sharing': 'teal',
'Security': 'green',
'VPN & Network': 'emerald',
'Dev: Editors': 'sky',
'Dev: Languages': 'rose',
'Dev: Tools': 'slate',
'Terminal': 'zinc',
'CLI Tools': 'gray',
'AI Tools': 'fuchsia'
};
function CategorySectionComponent({
category,
categoryApps,
selectedApps,
isAppAvailable,
selectedDistro,
toggleApp,
isExpanded,
onToggleExpanded,
focusedId,
focusedType,
onTooltipEnter,
onTooltipLeave,
categoryIndex,
onCategoryFocus,
onAppFocus,
isVerified,
getVerificationSource,
}: CategorySectionProps) {
const selectedInCategory = categoryApps.filter(a => selectedApps.has(a.id)).length;
const isCategoryFocused = focusedType === 'category' && focusedId === category;
const sectionRef = useRef<HTMLDivElement>(null);
const hasAnimated = useRef(false);
const prevAppCount = useRef(categoryApps.length);
const color = categoryColors[category] || 'gray';
useLayoutEffect(() => {
if (!sectionRef.current || hasAnimated.current) return;
hasAnimated.current = true;
const section = sectionRef.current;
const header = section.querySelector('.category-header');
const items = section.querySelectorAll('.app-item');
requestAnimationFrame(() => {
gsap.set(header, { clipPath: 'inset(0 100% 0 0)' });
gsap.set(items, { y: -15, opacity: 0, force3D: true });
const delay = categoryIndex * 0.05;
gsap.to(header, {
clipPath: 'inset(0 0% 0 0)',
duration: 0.6,
ease: 'power2.out',
delay: delay + 0.05
});
gsap.to(items, {
y: 0,
opacity: 1,
duration: 0.5,
ease: 'power2.out',
delay: delay + 0.1,
force3D: true
});
});
}, [categoryIndex]);
useEffect(() => {
if (categoryApps.length !== prevAppCount.current && sectionRef.current) {
const items = sectionRef.current.querySelectorAll('.app-item');
gsap.set(items, { y: 0, opacity: 1, clearProps: 'all' });
}
prevAppCount.current = categoryApps.length;
}, [categoryApps.length]);
return (
<div ref={sectionRef} className="mb-3 md:mb-5 category-section">
<CategoryHeader
category={category}
isExpanded={isExpanded}
isFocused={isCategoryFocused}
onToggle={() => {
const willExpand = !isExpanded;
onToggleExpanded();
if (willExpand) {
analytics.categoryExpanded(category);
} else {
analytics.categoryCollapsed(category);
}
}}
selectedCount={selectedInCategory}
onFocus={onCategoryFocus}
color={color}
/>
<div
className={`overflow-hidden transition-[max-height,opacity] duration-500 pt-0.5 ${isExpanded ? 'max-h-[2000px] opacity-100' : 'max-h-0 opacity-0'}`}
style={{ transitionTimingFunction: 'cubic-bezier(0.16, 1, 0.3, 1)' }}
>
{categoryApps.map((app) => (
<AppItem
key={app.id}
app={app}
isSelected={selectedApps.has(app.id)}
isAvailable={isAppAvailable(app.id)}
isFocused={focusedType === 'app' && focusedId === app.id}
selectedDistro={selectedDistro}
onToggle={() => toggleApp(app.id)}
onTooltipEnter={onTooltipEnter}
onTooltipLeave={onTooltipLeave}
onFocus={() => onAppFocus?.(app.id)}
color={color}
isVerified={
(selectedDistro === 'flatpak' || selectedDistro === 'snap') &&
isVerified?.(selectedDistro, app.targets?.[selectedDistro] || '') || false
}
verificationSource={
(selectedDistro === 'flatpak' || selectedDistro === 'snap')
? getVerificationSource?.(selectedDistro, app.targets?.[selectedDistro] || '') || null
: null
}
/>
))}
</div>
</div>
);
}
export const CategorySection = memo(CategorySectionComponent, (prevProps, nextProps) => {
if (prevProps.categoryApps.length !== nextProps.categoryApps.length) return false;
const prevIds = prevProps.categoryApps.map(a => a.id).join(',');
const nextIds = nextProps.categoryApps.map(a => a.id).join(',');
if (prevIds !== nextIds) return false;
if (prevProps.category !== nextProps.category) return false;
if (prevProps.isExpanded !== nextProps.isExpanded) return false;
if (prevProps.selectedDistro !== nextProps.selectedDistro) return false;
if (prevProps.focusedId !== nextProps.focusedId) return false;
if (prevProps.focusedType !== nextProps.focusedType) return false;
if (prevProps.categoryIndex !== nextProps.categoryIndex) return false;
if (prevProps.isVerified !== nextProps.isVerified) return false;
if (prevProps.getVerificationSource !== nextProps.getVerificationSource) return false;
for (const app of nextProps.categoryApps) {
if (prevProps.selectedApps.has(app.id) !== nextProps.selectedApps.has(app.id)) {
return false;
}
}
return true;
});