import React, { useCallback, useMemo, useRef } from 'react'; import { View, Text, StyleSheet, TouchableOpacity, Platform, Dimensions, FlatList } from 'react-native'; import { NavigationProp, useNavigation } from '@react-navigation/native'; import { useTranslation } from 'react-i18next'; 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, Layout } from 'react-native-reanimated'; import { RootStackParamList } from '../../navigation/AppNavigator'; import { getFormattedCatalogName, getCatalogDisplayName } from '../../utils/catalogNameUtils'; interface CatalogSectionProps { catalog: CatalogContent; } const { width } = Dimensions.get('window'); // Enhanced responsive breakpoints const BREAKPOINTS = { phone: 0, tablet: 768, largeTablet: 1024, tv: 1440, }; const getDeviceType = (deviceWidth: number) => { if (deviceWidth >= BREAKPOINTS.tv) return 'tv'; if (deviceWidth >= BREAKPOINTS.largeTablet) return 'largeTablet'; if (deviceWidth >= BREAKPOINTS.tablet) return 'tablet'; return 'phone'; }; const deviceType = getDeviceType(width); const isTablet = deviceType === 'tablet'; const isLargeTablet = deviceType === 'largeTablet'; const isTV = deviceType === 'tv'; // 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 { t, i18n } = useTranslation(); const navigation = useNavigation>(); const { currentTheme } = useTheme(); // Use state for the display name to handle async custom name resolution const [displayName, setDisplayName] = React.useState(catalog.name); // Re-resolve and format the name when language or catalog data changes React.useEffect(() => { const resolveName = async () => { // 1. Check for user-defined custom name const customName = await getCatalogDisplayName( catalog.addon, catalog.type, catalog.id, catalog.originalName || catalog.name ); // 2. If it's a user setting, use it as is if (customName !== (catalog.originalName || catalog.name)) { setDisplayName(customName); return; } // 3. Otherwise, use localized formatting const formatted = getFormattedCatalogName( customName, catalog.type, t('home.movies'), t('home.tv_shows'), t('home.channels') ); setDisplayName(formatted); }; resolveName(); }, [catalog.addon, catalog.id, catalog.type, catalog.name, catalog.originalName, i18n.language, t]); 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 (responsive spacing) const separatorWidth = isTV ? 12 : isLargeTablet ? 10 : isTablet ? 8 : 8; const ItemSeparator = useCallback(() => , [separatorWidth]); // Memoize the keyExtractor to prevent re-creation const keyExtractor = useCallback((item: StreamingContent) => `${item.id}-${item.type}`, []); return ( {displayName} navigation.navigate('Catalog', { id: catalog.id, type: catalog.type, addonId: catalog.addon }) } style={[ styles.viewAllButton, { paddingVertical: isTV ? 10 : isLargeTablet ? 9 : isTablet ? 8 : 8, paddingHorizontal: isTV ? 12 : isLargeTablet ? 11 : isTablet ? 10 : 10, borderRadius: isTV ? 22 : isLargeTablet ? 20 : isTablet ? 20 : 20, } ]} > {t('home.view_all')} ); }; const styles = StyleSheet.create({ catalogContainer: { marginBottom: 28, }, catalogHeader: { flexDirection: 'row', justifyContent: 'space-between', alignItems: 'center', marginBottom: 16, }, titleContainer: { position: 'relative', flex: 1, marginRight: 16, }, catalogTitle: { fontSize: 24, // will be overridden responsively fontWeight: '800', letterSpacing: 0.5, marginBottom: 4, }, titleUnderline: { position: 'absolute', bottom: -2, left: 0, width: 40, // overridden responsively height: 3, // overridden responsively borderRadius: 2, opacity: 0.8, }, viewAllButton: { flexDirection: 'row', alignItems: 'center', paddingVertical: 8, // overridden responsively paddingHorizontal: 10, // overridden responsively borderRadius: 20, // overridden responsively backgroundColor: 'rgba(255,255,255,0.1)', }, viewAllText: { fontSize: 14, // overridden responsively fontWeight: '600', marginRight: 4, // overridden responsively }, catalogList: { // padding will be applied responsively in JSX }, }); 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 ) ); });