import React, { useCallback } from 'react'; import { View, Text, StyleSheet, TouchableOpacity, Platform, Dimensions } from 'react-native'; import { FlashList } from '@shopify/flash-list'; import { NavigationProp, useNavigation } from '@react-navigation/native'; import { MaterialIcons } from '@expo/vector-icons'; import { LinearGradient } from 'expo-linear-gradient'; import { CatalogContent, StreamingContent } from '../../services/catalogService'; import { useTheme } from '../../contexts/ThemeContext'; import ContentItem from './ContentItem'; import Animated, { FadeIn } from 'react-native-reanimated'; import { RootStackParamList } from '../../navigation/AppNavigator'; interface CatalogSectionProps { catalog: CatalogContent; } const { width } = Dimensions.get('window'); // Dynamic poster calculation based on screen width - show 1/4 of next poster const calculatePosterLayout = (screenWidth: number) => { const MIN_POSTER_WIDTH = 100; // Reduced minimum for more posters const MAX_POSTER_WIDTH = 130; // Reduced maximum for more posters const LEFT_PADDING = 16; // Left padding const SPACING = 8; // Space between posters // Calculate available width for posters (reserve space for left padding) const availableWidth = screenWidth - LEFT_PADDING; // Try different numbers of full posters to find the best fit let bestLayout = { numFullPosters: 3, posterWidth: 120 }; for (let n = 3; n <= 6; n++) { // Calculate poster width needed for N full posters + 0.25 partial poster // Formula: N * posterWidth + (N-1) * spacing + 0.25 * posterWidth = availableWidth - rightPadding // Simplified: posterWidth * (N + 0.25) + (N-1) * spacing = availableWidth - rightPadding // We'll use minimal right padding (8px) to maximize space const usableWidth = availableWidth - 8; const posterWidth = (usableWidth - (n - 1) * SPACING) / (n + 0.25); if (posterWidth >= MIN_POSTER_WIDTH && posterWidth <= MAX_POSTER_WIDTH) { bestLayout = { numFullPosters: n, posterWidth }; } } return { numFullPosters: bestLayout.numFullPosters, posterWidth: bestLayout.posterWidth, spacing: SPACING, partialPosterWidth: bestLayout.posterWidth * 0.25 // 1/4 of next poster }; }; const posterLayout = calculatePosterLayout(width); const POSTER_WIDTH = posterLayout.posterWidth; const CatalogSection = ({ catalog }: CatalogSectionProps) => { const navigation = useNavigation>(); const { currentTheme } = useTheme(); const handleContentPress = useCallback((id: string, type: string) => { navigation.navigate('Metadata', { id, type, addonId: catalog.addon }); }, [navigation, catalog.addon]); const renderContentItem = useCallback(({ item }: { item: StreamingContent, index: number }) => { return ( ); }, [handleContentPress]); // Memoize the ItemSeparatorComponent to prevent re-creation const ItemSeparator = useCallback(() => , []); // Memoize the keyExtractor to prevent re-creation const keyExtractor = useCallback((item: StreamingContent) => `${item.id}-${item.type}`, []); return ( {catalog.name} navigation.navigate('Catalog', { id: catalog.id, type: catalog.type, addonId: catalog.addon }) } style={styles.viewAllButton} > View All {}} scrollEventThrottle={16} /> ); }; const styles = StyleSheet.create({ catalogContainer: { marginBottom: 28, }, catalogHeader: { flexDirection: 'row', justifyContent: 'space-between', alignItems: 'center', paddingHorizontal: 16, marginBottom: 16, }, titleContainer: { position: 'relative', flex: 1, marginRight: 16, }, catalogTitle: { fontSize: 24, fontWeight: '800', letterSpacing: 0.5, marginBottom: 4, }, titleUnderline: { position: 'absolute', bottom: -2, left: 0, width: 40, height: 3, borderRadius: 2, opacity: 0.8, }, viewAllButton: { flexDirection: 'row', alignItems: 'center', paddingVertical: 8, paddingHorizontal: 10, borderRadius: 20, backgroundColor: 'rgba(255,255,255,0.1)', }, viewAllText: { fontSize: 14, fontWeight: '600', marginRight: 4, }, catalogList: { paddingHorizontal: 16, }, }); export default React.memo(CatalogSection, (prevProps, nextProps) => { // Only re-render if the catalog data actually changes return ( prevProps.catalog.addon === nextProps.catalog.addon && prevProps.catalog.id === nextProps.catalog.id && prevProps.catalog.name === nextProps.catalog.name && prevProps.catalog.items.length === nextProps.catalog.items.length && // Deep compare the first few items to detect changes prevProps.catalog.items.slice(0, 3).every((item, index) => nextProps.catalog.items[index] && item.id === nextProps.catalog.items[index].id && item.poster === nextProps.catalog.items[index].poster ) ); });